-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathphotometryStep1AlignTraces_allbehav.m
216 lines (169 loc) · 9.26 KB
/
photometryStep1AlignTraces_allbehav.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
% takes in calcium data csv file (Neurophotometrics) and Noldus Observer manual scored behavior timesteamps
% puts them all on the same timescale for session start/stop
clear all
traces.M2 = 1;
traces.CStoM2 = 2;
traces.M2toCS = 3;
traces.CS = 4;
blackflyStimeFactor = 1000000000;
windowStart = 10;
%% import all data:
%name tracesTS, videoTS, tracesToD, videoToD, videoData, and tracesData
%Get photometry trace file
[filename, pathname] = uigetfile('*.csv', 'Select Traces Data File');
animalID = filename(1:4);
%Change to directory that all data is in
cd (pathname);
%Load traces data csv files
tracesData = dlmread(filename);
%Load tracesTS
[filename, pathname] = uigetfile('*.csv', 'Select Traces Timestamp File');
%Load csv files
tracestime = dlmread(filename);
tracesTS = tracestime(:,1);
tracesToD = tracestime(1,2);
%Load videoTS
[filename, pathname] = uigetfile('*.csv', 'Select Video Timestamp File');
%Load csv files
videotime = dlmread(filename);
videoTS = videotime(:,1);
videoToD = videotime(1,2);
%% check for dropped frames
plot(tracesTS)
userInput = input('Are there dropped frames? Press 1 for yes, 2 for no: ');
disp(userInput)
close all
if userInput == 1
dropFrameX = find(tracesTS==0);
dropOffset = ceil(median(diff(tracesTS)));
for x = 1:size(dropFrameX,1)
tracesTS(dropFrameX(x,1),1) = tracesTS((dropFrameX(x,1)-1),1) + dropOffset;
end
else
end
clear userInput dropFrameX dropOffset
%% Parse behavior data
[filename, pathname] = uigetfile('*.xlsx', 'Select Behavior Scoring Data File');
%Change to directory that all data is in
cd (pathname);
%Load csv files
videoxls = importdata(filename);
%Get list of behavior types and timestamps
behavior.type = videoxls.textdata(2:end, 12);
behavior.TS = videoxls.data(:, 6);
%Reduce timestamps to those that are *any* grooming
behavior.groomTS = behavior.TS(find(strcmpi(behavior.type, 'face grooming') | strcmpi(behavior.type, 'body grooming') | strcmp(behavior.type, 'scratching')));
behavior.groom = behavior.type(find(strcmpi(behavior.type, 'face grooming') | strcmpi(behavior.type, 'body grooming') | strcmp(behavior.type, 'scratching')));
behavior.rearTS = behavior.TS(find(strcmpi(behavior.type, 'rearing')));
%Parse apart grooming types
behavior.faceTS = behavior.groomTS(find(strcmpi(behavior.groom, 'Face Grooming')));
behavior.bodyTS = behavior.groomTS(find(strcmpi(behavior.groom, 'Body Grooming')));
behavior.scratchTS = behavior.groomTS(find(strcmpi(behavior.groom, 'Scratching')));
%Make a vector of groom type labels
groomtype = behavior.groom;
%% get session start frame as found in Noldus
sessionStartFrame = ceil(30*behavior.TS(find(strcmpi(behavior.type, 'LED on'))));
%% plot traces to determine order
plot(tracesData(1:3:end, 1)); %plot one third of M2 interleaved data in blue
hold on
plot(tracesData(2:3:end, 1)); %plot other third of M2 interleaved data in red
plot(tracesData(3:3:end, 1));
sigID = input('Which signal is the M2 gCamp? 1=blue, 2=red, 3=yellow');
gsig = sigID;
if gsig==1
rsig = 2;
bsig = 3;
elseif gsig==2
rsig = 3;
bsig = 1;
elseif gsig==3
rsig = 1;
bsig = 2;
end
close all
%% separate traces
%Get trace name
fnTraces = fieldnames(traces);
%Separate traces and assign to 'signal' for each ROI
for t=1:size(fnTraces,1)
signal.(fnTraces{t}).green = tracesData((gsig:3:end),t);
signal.(fnTraces{t}).red = tracesData((rsig:3:end),t);
signal.(fnTraces{t}).blue = tracesData((bsig:3:end),t);
%Plot all 3 channels in each ROI for visual inspection
figure
hold on
plot(signal.(fnTraces{t}).green);
plot(signal.(fnTraces{t}).red);
plot(signal.(fnTraces{t}).blue);
title(cellstr(fnTraces{t}));
end
%Separate timestamps by channel
tracesTSsep.green = tracesTS(gsig:3:end);
tracesTSsep.red = tracesTS(rsig:3:end);
tracesTSsep.blue = tracesTS(bsig:3:end);
%% zero timestamps and convert to seconds
fnChannels = fieldnames(tracesTSsep);
%Separate timestamps and convert to seconds and zeros to first timestamp
for c=1:size(fnChannels,1)
tracesTSzero.(fnChannels{c}) = timestampDecoder(tracesTSsep.(fnChannels{c}))'; %run Blackfly FP camera timestamps through timestampDecoder function; converts binary cycled numbers into seconds and zeros it to first timestamp
end
videoTSzero = ((videoTS-videoTS(1,1))/blackflyStimeFactor)'; %convert Blackfly S camera timestamps from nanoseconds to seconds and zero to first timestamp
tracesToDsec = tracesToD/1000; %convert first Bonsai time-of-day timestamp for Blackfly FP camera to seconds
videoToDsec = videoToD/1000; %convert first Bonsai time-of-day timestamp for Blackfly S camera to seconds
%% align timestamps by using first Bonsai timestamps
for c=1:size(fnChannels,1)
tracesTSToD.(fnChannels{c}) = tracesTSzero.(fnChannels{c}) + tracesToDsec; %add first Bonsai Blackfly FP ToD timestamp to whole series; puts timestamps on same time scale
end
videoTSToD = videoTSzero + videoToDsec; %add first Bonsai Blackfly S ToD timestamp to whole series; puts timestamps on same time scale
%% re-zero timestamps based on session start
for c=1:size(fnChannels,1)
subTS.traces.(fnChannels{c}) = (tracesTSToD.(fnChannels{c}) - (videoTSToD(1,sessionStartFrame))); %subtract the value of the corresponding start frame in the video timestamp vector from all values of the traces timestamp series
end
subTS.video = (videoTSToD - (videoTSToD(1,sessionStartFrame))); %subtract the value of the corresponding start frame in the video timestamp vector from all values of the video timestamp series
%add to all behavior timestamps
behavior.sub.face = (behavior.faceTS + subTS.video(1,1))'; %add first value of properly aligned video timestamp to video time series from Noldus to transform to proper time scale
behavior.sub.body = (behavior.bodyTS + subTS.video(1,1))';
behavior.sub.scratch = (behavior.scratchTS + subTS.video(1,1))';
behavior.sub.groom = (behavior.groomTS + subTS.video(1,1))';
behavior.sub.rear = (behavior.rearTS + subTS.video(1,1))';
%% separate Noldus timestamps into behavior start and end categories
%Get last timestamp of traces, subtract window to find latest grooming stop
sessionEnd = subTS.traces.green(end);
windowEnd = sessionEnd-windowStart;
%Go through each type of behavior and crop timestamps to be within session time
fnSub = fieldnames(behavior.sub);
for s=1:(size(fnSub,1))
temp.start = behavior.sub.(fnSub{s})(1,1:2:end); %put all start timestamps into temp.start; it will be every other timestamp starting with the first
temp.stop = behavior.sub.(fnSub{s})(1,2:2:end); %put all end timestamps into temp.end; it will be every other timestamp starting with the second
%get fieldnames of temp (start, end)
fnTemp = fieldnames(temp);
%cycle through start and stop timestamps
for i = 1:size(fnTemp,1)
%trim the timestamps to that they're within the window of interest (windowStart and windowEnd)
behavAlign.(fnSub{s}).(fnTemp{i}) = temp.(fnTemp{i})(find(temp.(fnTemp{i}) >= windowStart & temp.(fnTemp{i})<= windowEnd));
end
if size(behavAlign.(fnSub{s}).start,2) > size(behavAlign.(fnSub{s}).stop,2) %exclude hanging start timestamp that doesn't have matching end; it means end was outside of range, and it should be excluded from further analysis
behavAlign.(fnSub{s}).start = behavAlign.(fnSub{s}).start(1,1:end-1);
elseif size(behavAlign.(fnSub{s}).start,2) < size(behavAlign.(fnSub{s}).stop,2) %exclude first stop timestamp that deosn't have matching start; it means beginning was outside of range, and it should be excluded from furhter analysis
behavAlign.(fnSub{s}).stop = behavAlign.(fnSub{s}).stop(1,2:end);
end
end
%% crop camera timestamps to session start and end
%cycle through subTS fieldnames (traces and video) to create new vectors with only the values between the session start and end
for c=1:size(fnChannels,1)
idx.traces.(fnChannels{c}) = subTS.traces.(fnChannels{c})>= 0 & subTS.traces.(fnChannels{c})<= sessionEnd;
alignTS.traces.(fnChannels{c}) = subTS.traces.(fnChannels{c})(idx.traces.(fnChannels{c}));
end
idx.video = subTS.video >= 0 & subTS.video<= sessionEnd;
alignTS.video = subTS.video(idx.video);
%% crop traces data to session start and end
fnTraces = fieldnames(signal); %get fieldnames of signal
for t = 1:size(fnTraces,1) %cycle through fnTrace fieldnames to create new vectors with values between session start and end
for c=1:size(fnChannels,1)
alignTraces.(fnTraces{t}).(fnChannels{c})= signal.(fnTraces{t}).(fnChannels{c})(idx.traces.(fnChannels{c})); %use traces index from timestamp alignment to create new vectors
end
end
%% clear variables and save data
clearvars -except alignTS alignTraces behavAlign animalID
save([animalID, '-alignedDetailedBehav.mat']);
clear all