-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathetra_challenge.qmd
958 lines (748 loc) · 30.3 KB
/
etra_challenge.qmd
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
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
---
title: "ETRA Challenge Report"
author: "Dávid Kubek"
date: last-modified
date-format: long
bibliography: references.bib
format:
html:
toc: true
theme: custom.scss
code-fold: true
code-line-numbers: true
pdf:
echo: false
execute:
cache: true
freeze: true
jupyter: python3
---
In 2019, the [ETRA](https://etra.acm.org/) organization announced a challenge
for the analysis of a dataset pertaining to human eye-movement. The objective of
the challenge was to utilize various tools to uncover intriguing relationships
and information within the data, potentially leading to novel insights.
Additional details regarding the challenge can be found in the official
[Challenge Track Call for Papers](https://etra.acm.org/2019/challenge.html)
issued by ETRA.
```{python}
#| code-summary: Imports and setup
#| output: false
import warnings
warnings.simplefilter(action='ignore', category=FutureWarning)
# Import libraries
import pandas as pd
import numpy as np
import scipy.stats as stats
import pymc as pm
import seaborn as sns
import matplotlib.pyplot as plt
import arviz as az
from etra import read_data
# Apply the default theme
sns.set_theme()
```
# Getting data
Download zip file of the [dataset](http://smc.neuralcorrelate.com/ETRA2019/ETRA2019Challenge.zip). Description of the dataset and how
the data was collected can be found on [ETRA dataset description](https://etra.acm.org/2019/challenge.html).
```{python}
#| code-summary: Download data
#| output: false
from etra import ETRA
dataset = ETRA()
```
The directory ``data/`` contains the following directories/files:
- ``data/`` : Contains subdirectories containing data of individual
participants. Each subdirectory is filled with CSV files, each containing 45
seconds of eye tracker data
- ``images/`` : Contains the images shown to the participants for each task.
- ``DataSummary.csv`` : Contains mouse clicks of the participants for tasks like
Puzzle or Where's Waldo.
# Hypotheses
We will be interested in the following hypotheses:
1. **Is there a point where a participant becomes tired/bored when viewing a scene? (the frequency of saccades decreases) If so, are some scenes less interesting then others?**
There are different types of scenes in the study and some might be more
visually stimulating then others. For example, it is not unreasonable to
expect that a participant will stay much more engaged in a scene with a lot
of detail to study then when looking at a blank scene. Another possibility
is that during visual searches (picture puzzles, _Where is Waldo_) a
participant may become frustrated and just give up on the task.
The reasoning behind the way we are going to measure this is as follows.
When a participant is engaged in a scene, we assume that their gaze is going
to dart across the scene rapidly, as they are studying all the details and
the number of saccades during a fixed time frame is going to be very high.
On the other hand, when subject becomes bored or frustrated, we might expect
that they are going to rest their gaze at some point and not move their eyes
very much (maybe just waiting for the trial to be over) and so the frequency
of saccades would decrease.
However, the decrease in frequency of saccades might not be exactly
an indicator of boredom or frustration. It is not unlikely that
the frequency of saccades is high every time a new scene is introduced and
naturally decreases as the subject becomes acquainted with the scene.
Whatever the reason might be, if there indeed is such a changing point in
the frequency of saccades, it might be interesting to study the time for
such change and it's relationship to participant, scene and task.
2. **Does the pupil size change depending on scene? What about dependence on task?**
The motivation behind this hypothesis is that we would like to study the
excitement of a subject viewing some scene or performing some task. One of
the indicators of excitement might be the change in pupil size (i.e. we
expect pupil to dilate during excitement and contract otherwise).
There are various tasks and scenes and the level of excitement (pupil
dilation) might differ in each:
* in **picture puzzles** we might expect that the level of excitement is
kept high during the whole duration, peaking when subject finds a difference,
* in _**Where is Waldo?**_ puzzle we might expect the level of excitement
being low as the subject is having difficulties finding Waldo and peaking
when (if) Waldo is found,
* viewing a **blank scene** might not excite much.
Therefore, it is interesting to ask which tasks and scenes are the most
visually stimulating and whether any significant distinction can be made.
```{python}
#| code-summary: Choose an example subject
subject_no = 22
```
# Hypothesis 1: Saccade Frequency
We will restrict ourselves only to the following hypothesis:
* $H_0$ _(null hypothesis)_ : The frequency of saccades does not decrease when
viewing a scene.
* $H_1$ _(alternative hypothesis)_ : The frequency of saccades decreases at some
point in the trial.
## Load Data
We will restrict ourselves only to one subject. We have arbitrarily chosen the
participant with ID ``022``. We select and load all _Free Viewing_ data with
``Natural`` and ``Blank`` scenes. We will be interested in columns:
* ``time`` : Indicating the time in milliseconds from the beggining of the trail.
* ``x``, ``y`` : The position on the screen where the participant was looking at the given time.
* ``rp``: Pupil size of the right eye.
In @tbl-hyp1-data we provide a sample of the data we will use.
```{python}
#| code-summary: Load data
df_hyp1 = (dataset.data_dir / "data" / "{0:0>3}".format(subject_no))\
.glob("*FreeViewing_*.csv")
df_hyp1 = pd.concat((read_data(f) for f in df_hyp1)).sort_values(by="Time")
df_hyp1 = df_hyp1.rename(
{
"Time": "time",
"trial_id": "trial",
"LXpix": "x",
"LYpix": "y",
"RP": 'rp'
},
axis=1
)
df_hyp1["time"] = df_hyp1.groupby(["participant_id", "trial"])["time"]\
.transform(lambda x: x - x.min())
df_hyp1 = df_hyp1[
[
'participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id',
'time', 'x', 'y', 'rp'
]
]
```
```{python}
#| code-summary: Print sample of hypothesis 1 data
#| label: tbl-hyp1-data
#| tbl-cap: Sample of the data for hypothesis 1.
#| warning: false
df_hyp1.head()
```
## Data Manipulation
First step of the pre-processing will be to remove the blinks. We look at the
pupil size and consider every value smaller than some threshold to be an
indication of a blink. We take the average pupil to be between 1 and 8 mm
[@PupilDiameter]. We set 1 mm as our threshold. We will then discard a portion
of our data around such points. The average blink duration ranges between 100
and 400 ms [@BlinkDuration]. We delete 200 ms before and after each point that
falls below the threshold and another 200 ms before and after each
blink/semi-blink to eliminate the initial and final parts in which the pupil was
still partially occluded.
```{python}
#| code-summary: Function for removing blinks
def remove_blinks(data : pd.DataFrame, column='rp'):
ans = data.copy()
n_rows = ans.shape[0]
lows = np.where(ans['rp'].lt(100))[0]
for low in lows:
ans.loc[max(0, low-40):min(low+40, n_rows), column] = pd.NA
return ans.dropna()
```
```{python}
#| label: fig-blinks-removed
#| fig-cap: Pupil size data with blinks (top) and with blinks removed (bottom).
#| code-summary: Plot pupil size with and without blinks
before = df_hyp1[df_hyp1.trial == '106'].assign(blinks_removed='No')
after = remove_blinks(before).assign(blinks_removed='Yes')
hlp = pd.concat([before, after])
g = sns.relplot(
data=hlp,
x='time', y='rp',
kind='line',
row='blinks_removed',
aspect=3
)
g.set_xlabels("Time (ms)")
g.set_ylabels("Pupil size")
g.axes[0, 0].set_title("Pupil size as measured")
g.axes[1, 0].set_title("Pupil size with blinks removed");
```
For each trial, we will detect saccades using the [REMoDNaV](https://github.com/psychoinformatics-de/remodnav)
python library [@Dar2019]. This gives us the start and end times of saccades
(and other events) as well as additional information such as start and end
position of the gaze or peak velocity. We will consider only the start times of
the saccades and disregard all other data. @tbl-natural-sacc-data lists a sample
of the processed data.
```{python}
#| output: false
#| code-summary: Function for detecting saccades
from etra import detect
def detect_saccades_by_groups(data, groupby=["participant_id", "trial"]):
ans = []
groups = data.groupby(groupby)
for (pid, trial), group in groups:
tmp = remove_blinks(group)
tmp = detect(group)
tmp = tmp[tmp["label"] == "SACC"]
tmp["participant_id"] = pid
tmp["trial"] = trial
ans.append(tmp)
return pd.concat(ans)
```
```{python}
#| code-summary: Preprocess data
#| warning: false
df_hyp1_natural = df_hyp1[df_hyp1.task_type == "Natural"]
df_hyp1_natural_sacc = detect_saccades_by_groups(
df_hyp1_natural
)
df_hyp1_blank = df_hyp1[df_hyp1.task_type == "Blank"]
df_hyp1_blank_sacc = detect_saccades_by_groups(
df_hyp1_blank
)
trial_info_natural = df_hyp1_natural[
['participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id']
].drop_duplicates()
df_hyp1_natural_sacc = trial_info_natural.merge(df_hyp1_natural_sacc, on=[
"participant_id", "trial"], how="left")
df_hyp1_natural_sacc = df_hyp1_natural_sacc[
[
'participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id',
'start_time',
]
]
trial_info_blank = df_hyp1_blank[
['participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id']
].drop_duplicates()
df_hyp1_blank_sacc = trial_info_blank.merge(df_hyp1_blank_sacc, on=[
"participant_id", "trial"],
how="left"
)
df_hyp1_blank_sacc = df_hyp1_blank_sacc[
[
'participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id',
'start_time',
]
]
```
```{python}
#| code-summary: Print sample of processed data
#| label: tbl-natural-sacc-data
#| tbl-cap: Sample of Freeviewing data.
#| warning: false
df_hyp1_natural_sacc.head()
```
## Methodology and Exploration
To help us understand how we should model the observations, we first look at the
distribution of saccades in time. The $x$ axis represents the time from the
start of a trial and each row is a sequence of dots representing the start time
of a saccade.
```{python}
#| label: fig-saccades-in-time
#| fig-cap: Distribution of saccades in time in Natural scenes for one participant.
#| code-summary: Plot the distriution of saccades in time
g = sns.catplot(
data=df_hyp1_natural_sacc,
x='start_time',
y='stimulus_id',
jitter=False,
aspect=3,
)
g.set_xlabels("Start time of a saccade in milliseconds")
g.set_ylabels("ID of the stimulus")
g.set(title="Distribution of saccades in time");
```
As the occurance of a saccade is a complex process that depends on the scene
viewed, mental state of the participant, tiredness or maybe even things like
personal history, eyesight, mood and many other factors, it is impossible for us
to accurately predict the occurance of the next saccade. To simplify the
situation, we might look at it as on a stochastic process, where for some small
enough time interval, we have a fixed probability for an occurance of a saccade.
Let $N_t$ be a variable counting the number of saccades in a time interval
$[0,t]$. Our assumption can be formulated as
$$
\lim_{t \to 0} \Pr[N_t \ge 1]/t = \lambda
$$
for some $\lambda \in \mathbb{R}^+$. That is, the probability of an saccade in a
small time interval $[0, t]$ tends to $\lambda t$.
Due to physical human limitations, we also have
$$
\lim_{t \to 0} \Pr[N_t \ge 2]/t = 0
$$
or said differently, for a small enough time interval, it is impossible to have
two or more successive saccades.
The number of saccades in some interval $[t, s]$ can be expressed then as $N(t)
- N(s)$. Additionally, we will assume that for any two disjoint time intervals
$[t_1, t_2]$ and $[t_3, t_4]$, the distribution of $N(t_2) - N(t_1)$ is
independent of the distribution of $N(t_4) - N(t_3)$.
With these assumtions in place, we are justified to model the occurances of
saccades as a _Poisson process_ with rate $\lambda$ [@mitzenmacher_upfal_2012].
```{python}
#| code-summary: Function for counting the number of saccades each second
def saccade_frequency_count_by_second(data):
""" Count the number of saccades in each second of the trial """
bins = (data.start_time // 1000).value_counts().sort_index()
data = pd.DataFrame({"time": range(45)})
data["sacc_count"] = bins
data.fillna(0, inplace=True)
data.sacc_count = data.sacc_count.astype(int)
return data
```
The theory of Poisson processes then tells us that the number of saccades in
some time period of length $t$ is a Poisson random variable with expectation
$\lambda t$. We will divide the time of the experiment to seconds (so $t = 1$)
and count the number of saccades in each second. The @fig-saccade-dist
illustrates such counts for one chosen trial.
```{python}
#| code-summary: Chose one scene as an example
stimulus = "nat013"
scene = df_hyp1_natural_sacc[df_hyp1_natural_sacc.stimulus_id == stimulus]
```
```{python}
#| code-summary: Plot the frequency of saccades in time
#| label: fig-saccade-dist
#| fig-cap: Frequency of saccades in time for one chosen natural scene.
hlp = scene.copy()
hlp.start_time = hlp.start_time.apply(lambda ms: ms // 1000)
g = sns.displot(data=hlp, x='start_time', bins=45, stat="count", aspect=3)
g.set_xlabels("Time (s)")
g.set_ylabels("Number of saccades")
g.set(title=f"Frequency of saccades");
```
We are looking for a change in behaviour in our participants. To model it we
will assume that in some initial time interval the number of saccades will
follow some Poisson process with rate $\lambda_1$ up until some time $\tau$
after which _something_ changes and the saccades will follow a different process
with rate $\lambda_2$.
Let $S_t$ denote the count of saccades at time $t$. We have
$$
S_t \sim \mathrm{Poisson}(\lambda^{(t)})
$$
However, we do not know what the parameter $\lambda^{(t)}$ is. It might be either
$\lambda_1$ or $\lambda_2$
depending on where we put the changing point $\tau$, so
$$
\lambda^{(t)}
= \begin{cases}
\lambda_1 & t < \tau \\
\lambda_2 & t \ge \tau
\end{cases}
$$
We will try to infer these variables from our data using Bayesian statistics. We
will need to choose prior distributions to do that. For $\lambda_1$ and
$\lambda_2$ we need to choose from continouous distribution that is able to
obtain (potentially) all positive values. Exponential distribution
satisfies this requirement, so we set
$$
\lambda_1 \sim \mathrm{Exp}(\alpha)
$$
$$
\lambda_2 \sim \mathrm{Exp}(\alpha)
$$
where $\alpha$ is some hyper-parameter. As we have no prior knowledge about
$\tau$, we simply choose
$$
\tau \sim \mathrm{Unif}(0, 44)
$$
To help with the inference, we set the hyperparameter $\alpha$ such that
$$
\frac{1}{\alpha}
= \mathbb{E}[\lambda \mid \alpha]
\approx \frac{1}{45} \sum_{t = 0}^{44} S_t
$$
which is _about_ the value we would expect the $\lambda$s to be distributed
around.
::: {.callout-note}
This approach was adapted from the first chapter of
[Bayesian Methods for Hackers](https://github.com/CamDavidsonPilon/Probabilistic-Programming-and-Bayesian-Methods-for-Hackers).
:::
For our modelling we will use the [PyMC](https://www.pymc.io/welcome.html)
[@Salvatier2016] implementation of Monte Carlo Markov Chains, using the NUTS
sampler with default parameters to generate $5000$ samples from the posterior
distribution.
```{python}
#| code-summary: Define model for the saccade frequency
def model_saccades(data):
# Get the count data for one chosen trial
count_data = saccade_frequency_count_by_second(data)
count_data = count_data.sacc_count
n_count_data = count_data.shape[0]
with pm.Model() as model:
alpha = 1.0/count_data.mean()
lambda_1 = pm.Exponential("lambda_1", alpha)
lambda_2 = pm.Exponential("lambda_2", alpha)
tau = pm.DiscreteUniform("tau", lower=0, upper=n_count_data - 1)
index = np.arange(n_count_data)
lambda_ = pm.math.switch( tau >= index, lambda_1, lambda_2 )
delta = pm.Deterministic("delta", lambda_2 - lambda_1)
observation = pm.Poisson("obs", lambda_, observed=count_data)
return model
```
```{python}
#| code-summary: Run the chain
#| output: false
model = model_saccades(scene)
with model:
trace = pm.sample(5_000)
```
@fig-example-chain-posterior shows the posterior distributions of the
parameters $\tau$, $\lambda_1$, $\lambda_2$ and $\Delta = \lambda_2 - \lambda_1$,
while @tbl-example-chain-summary provides summary statistics from the chain.
```{python}
#| label: fig-example-chain-posterior
#| fig-cap: The posterior distribution of the model parameters.
#| code-summary: Plot the posterior of model parameters
az.plot_posterior(trace);
```
```{python}
#| label: tbl-example-chain-summary
#| tbl-cap: Summary statistics for the example chain.
#| code-summary: Print summary statistics of the chain
#| warning: false
az.summary(trace)
```
Notice that the posterior distribution of $\Delta$ is centered around 0, leading
us to believe there is no change in the the rate of saccades. In case there was
a change, we would expect a shift either to the positive or negative side. This
notion is further reinforced by looking at the Highest Density Interval (HDI) of
the posterior distribution of $\tau$ which basically spans the entire duration
of the trial.
```{python}
#| code-summary: Calculate the probability of null hypothesis
p_l2_ge_l1 = np.mean(trace.posterior["delta"].values >= 0)
```
We have
```{python}
#| echo: false
print("Pr[H_0] = Pr[lambda_2 >= lambda_1]= {:0.3}".format(p_l2_ge_l1))
```
And as the probability is quite high, we do not reject the null hypothesis in
this example.
### Natural scenes
We do the same analysis as above for all trials using _Natural_ scenes for
subject ``022``.
```{python}
#| code-summary: Sample the posterior distributions for all natural scenes
#| output: false
stimuli_natural = df_hyp1_natural_sacc.stimulus_id.unique()
stimuli_natural.sort()
chains_natural = []
for stimulus in stimuli_natural:
trial = df_hyp1_natural_sacc[df_hyp1_natural_sacc.stimulus_id == stimulus]
model = model_saccades(trial)
with model:
chain = pm.sample(5_000)
chains_natural.append(chain)
```
```{python}
#| code-summary: Combine Natural data from all chains
combined_natural = []
for stimulus, chain in zip(stimuli_natural, chains_natural):
df = chain.posterior.to_dataframe()
df["stimulus"] = stimulus
combined_natural.append(df)
combined_natural = pd.concat(combined_natural)
combined_natural = combined_natural.assign(abs_delta=lambda _: np.abs(_['delta']))
```
@fig-natural-delta-posterior shows the posterior density estimate for $\Delta$
for all _Natural_ scene trials.
```{python}
#| code-summary: Plot Posterior density of $\Delta$ for all natural scenes.
#| label: fig-natural-delta-posterior
#| fig-cap: Posterior density of $\Delta$ for all natural scenes of one chosen subject.
statistic = "delta"
# Truncate values to 99% most plausible
low, up = combined_natural.delta.quantile([0.005, 0.995])
hlp = pd.DataFrame()
hlp[statistic] = combined_natural[statistic].where(
combined_natural[statistic].between(low, up)
)
hlp["stimulus"] = combined_natural["stimulus"]
hlp.reset_index(inplace=True)
g = sns.displot(data=hlp, x=statistic, hue="stimulus", kind="kde", aspect=2)
g.ax.axvline(0, color='k', ls='dashed', alpha=0.5)
g.set_xlabels("$\Delta$")
g.set_ylabels("Probability")
g.set(title=f"Posterior density of $\Delta$");
```
There does seem to be some variability from scene to scene, however, from the
graph we can still see that _on average_ the change of the frequencies seems to
be centered around zero, even shifted sligtly toward the negative side meaning
that it is more probable that the frequency of saccades _decreases_ slightly as
time progresses.
### Blank scenes
We now repeat the exact same procedure, but for trials with _Blank_ scene.
```{python}
#| code-summary: Sample the posterior distributions for all blank scenes
#| output: false
trials_blank = df_hyp1_blank_sacc.trial.unique()
trials_blank.sort()
chains_blank = []
for trial in trials_blank:
trial_data = df_hyp1_blank_sacc[df_hyp1_blank_sacc.trial == trial]
model = model_saccades(trial_data)
with model:
chain = pm.sample(5_000)
chains_blank.append(chain)
```
```{python}
#| code-summary: Combine Blank data from all chains
combined_blank = []
for trial, chain in zip(trials_blank, chains_blank):
df = chain.posterior.to_dataframe()
df["trial"] = trial
combined_blank.append(df)
combined_blank = pd.concat(combined_blank)
combined_blank = combined_blank.assign(abs_delta=lambda _: np.abs(_['delta']))
```
```{python}
#| code-summary: Plot posterior density of $\Delta$ for all blank scenes.
#| label: fig-blank-delta-posterior
#| fig-cap: Posterior density of $\Delta$ for all blank scenes of one chosen subject.
statistic = "delta"
# Truncate values to 99% most plausible
low, up = combined_blank.delta.quantile([0.005, 0.995])
hlp = pd.DataFrame()
hlp[statistic] = combined_blank[statistic].where(
combined_blank[statistic].between(low, up)
)
hlp["trial"] = combined_blank["trial"]
hlp.reset_index(inplace=True)
g = sns.displot(data=hlp, x="delta", hue="trial", kind="kde", aspect=2)
g.ax.axvline(0, color='k', ls='dashed', alpha=0.5);
```
```{python}
#| code-summary: Compute probabilities of extreme cases
p_l2_lt_l1_094 = (combined_blank[combined_blank.trial == "094"].delta < 0).mean()
p_l2_gt_l1_070 = (combined_blank[combined_blank.trial == "070"].delta > 0).mean()
```
The situation here is much more diverse in this case. In
@fig-blank-delta-posterior we can see that we are able to find trials on both
ends of the spectrum when considering the change in frequency of saccades. For
example, for the trial ``094`` we have
```{python}
#| echo: false
print("Pr[ lambda_2 < lambda_1 ] = {}".format(p_l2_lt_l1_094))
```
so there is a very high probability of a _decrease_ in the frequency of saccades
at some point in the trial.
On the other hand, for trial ``061`` we have
```{python}
#| echo: false
print("Pr[ lambda_2 > lambda_1 ] = {}".format(p_l2_gt_l1_070))
```
meaning a high probability of an _increase_ in frequency.
This seems quite surprising as one might expect consistent behaviour when
viewing the same blank scene multiple times.
## Results
```{python}
#| code-summary: Compute the probability of null hypothesis being true
p_l2_ge_l1_natural_combined = (combined_natural.delta >= 0).mean()
```
Considering all natural scenes as a whole, the probability that the frequency of
saccades does not decrease is
```{python}
#| echo: false
print("Pr[ H_0 ] = Pr[ lambda_2 >= lambda_1 ] = {:0.3}".format(
p_l2_ge_l1_natural_combined)
)
```
which is not low enough to reject the null hypothesis
```{python}
#| code-summary: Compute probability of null hypothesis for Blank trials
p_l2_ge_l1_blank_combined = (combined_blank.delta >= 0).mean()
```
Considering all the blank trials as a whole, the probability of increase in
saccades is
```{python}
#| echo: false
print("Pr[ H_0 ] = Pr[ lambda_2 >= lambda_1 ] = {:0.3}".format(
p_l2_ge_l1_blank_combined)
)
```
which is not low enough to reject the null hypothesis
## Discussion
In this section, we will examine certain assumptions that may have influenced
the outcome of our model. One assumption made is that the distribution of
saccades is independent of the selected time interval for examination. However,
it is possible that the number of saccades may vary over time due to factors
such as participant fatigue.
Additionally, we have assumed that there is a single point of change within the
data. However, it is feasible that there may be multiple points of change or
that the change may occur gradually. Further exploration of these possibilities
may serve as a means for model improvement.
Furthermore, our model utilizes specific priors for the parameters. While we
have chosen what we believe to be conservative priors, alternative options such
as Half-Cauchy or Truncated Normal distributions for the $\lambda$s should also
be considered. Additionally, the selection of hyperparameters should be
evaluated. We claim that the choice of hyperparameters did not have a
significant impact on the model's results and served primarily to improve the
speed of convergence of the chain.
Lastly, we did not take into account the examination of trials as a cohesive
unit. Each trial was analyzed individually, but it is possible that the order,
type and difficulty of prior trials may have affected performance on subsequent
trials.
# Hypothesis 2: Pupil dilation
We will study the following hypotheses:
* $H_0^{(T)}$ _(null hypothesis)_ : The mean pupil size is equal when comparing
Blank scene and a type T scene
* $H_1^{(T)}$ _(alternative hypothesis)_ : The mean pupil size is _not_ equal
when comparing Blank scene and a type T scene
## Load Data
Again, we will limit our analysis to data collected from a single participant,
specifically participant ID ``022``. The participant was selected arbitrarily.
We have selected and loaded all data collected during the "Free Viewing" task.
Our focus will be on the following columns of the dataset:
* ``time``: representing the time in milliseconds elapsed from the beginning of
the trial.
* ``rp``: indicating the size of the right pupil.
In @tbl-hyp2-data we list a sample of data we work with.
```{python}
#| code-summary: Load data
#| output: false
df_hyp2 = (dataset.data_dir / "data" / "{0:0>3}".format(subject_no))\
.glob("*FreeViewing_*.csv")
df_hyp2 = pd.concat((read_data(f) for f in df_hyp2)).sort_values(by="Time")
df_hyp2 = df_hyp2.rename(
{
"Time": "time",
"trial_id": "trial",
"RP": "rp"
},
axis=1
)
df_hyp2["time"] = df_hyp2.groupby(["participant_id", "trial"])["time"]\
.transform(lambda x: x - x.min())
df_hyp2 = df_hyp2[
[
'participant_id', 'trial', 'fv_fixation', 'task_type', 'stimulus_id',
'time','rp',
]
]
```
```{python}
#| code-summary: List a sample of data for hypothesis 2.
#| label: tbl-hyp2-data
#| tbl-cap: Sample of data for hypothesis 2
#| warnings: false
df_hyp2.head()
```
## Data Manipulation
Again, we remove the blinks and then calculate the mean pupil size for each
trial.
```{python}
#| code-summary: Remove blinks and calculate average pupil size.
df_hyp2_avg_rp = df_hyp2.groupby("trial")\
.apply(remove_blinks)\
.reset_index(drop=True)\
.groupby(["trial", "task_type"])\
.agg(avg_rp=('rp', 'mean'))\
.reset_index()\
.drop("trial", axis=1)
```
```{python}
#| code-summary: Report the average pupil size for each task type.
#| label: tbl-avg-rp
#| tbl-cap: Average pupil size (right eye) for each task type.
#| warning: false
df_hyp2_avg_rp.head().round(decimals=2)
```
## Exploration
Looking at @fig-avg-rp-dist-box, we can see that there except for the "Natural"
scenes there are no significant outliers in any other category.
@tbl-avg-rp-summary lists summary statistics about the average pupil size.
```{python}
#| code-summary: Plot the distribution of the average pupil size.
#| label: fig-avg-rp-dist-box
#| fig-cap: Distribution of the average pupil size across different task types.
g = sns.catplot(
data=df_hyp2_avg_rp,
x='task_type', y='avg_rp',
kind='box',
aspect=3
)
g.set_xlabels("Task type")
g.set_ylabels("Average pupil size")
g.set(title="Average pupil size distribution.");
```
```{python}
#| code-summary: Compute summary statistics for average pupil sizes
#| label: tbl-avg-rp-summary
#| tbl-cap: Summary statistics for average pupil sizes.
#| warnings: false
df_hyp2_avg_rp.groupby("task_type").describe().round(decimals=2)
```
Let us look at the distribution of the means. If we want to use t-test, we want
our data to be roughly normally distributed.
```{python}
#| code-summary: Plot the distribution of average pupil sizes
#| label: fig-avg-rp-dist-bar
#| fig-cap: Distribution of average pupil sizes by task type.
g = sns.displot(
data=df_hyp2_avg_rp,
x='avg_rp',
hue='task_type',
col="task_type"
)
g.set_xlabels("Average pupil size")
g.set_ylabels("Count");
```
Looking at the graphs in @fig-avg-rp-dist-bar, the distribution of the data
seems to be roughly normal, however it is hard to gauge this from such a small
sample size. We use the Shapiro-Wilk test, which tests the null hypothesis that
the data was drawn from a normal distribution.
```{python}
#| code-summary: Compute Shapiro-Wilk test for distribution normality
print("Resulting p-values from the Shapiro-Wilk test:\n")
for task_type in ["Blank", "Puzzle", "Waldo", "Natural"]:
shapiro_result = stats.shapiro(
df_hyp2_avg_rp[df_hyp2_avg_rp.task_type == task_type].avg_rp
)
print("\t{}\t: {:0.3}".format(task_type, shapiro_result.pvalue))
```
Using signifance level of $\alpha = 0.05$, we conclude that the data is
distributed normally (although this is a bit debatable in the case of the
"Puzzle" task).
## Results
Given that we are evaluating the same participant across different tasks, it is
appropriate to use the paired t-test as the data points are linked to one
another. This statistical test allows for the comparison of each task with our
chosen baseline, which is the blank scene.
```{python}
#| code-summary: Compute paired t-test for each category with the baseline.
baseline = "Blank"
print("Resulting p-values from the paired t-test:\n")
for task_type in ["Puzzle", "Waldo", "Natural"]:
ttest_result = stats.ttest_ind(
df_hyp2_avg_rp[df_hyp2_avg_rp.task_type == baseline].avg_rp,
df_hyp2_avg_rp[df_hyp2_avg_rp.task_type == task_type].avg_rp,
)
print("\t{}\t: {:0.3}".format(task_type, ttest_result.pvalue))
```
Thus using significance level of $\alpha = 0.05$, we reject the null hypothesis
for the _Puzzle_ and _Waldo_ tasks, meaning the mean pupil size differs from the
baseline (greater as can be seen from @fig-avg-rp-dist-box). We were not able to
reject the null hypothesis in case of the _Natural_ task.
## Discussion
Another interesting metric to consider might be the number of times the pupil
size peaks and the count of such peaks.
# References
::: {#refs}
:::