-
Notifications
You must be signed in to change notification settings - Fork 0
/
article_formatted_doc.Rmd
1793 lines (1158 loc) · 103 KB
/
article_formatted_doc.Rmd
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
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
---
title: "Mission Unaccomplished: A Funding Programme for Early Career Researchers Fails to Promote Independence"
output: bookdown::word_document2
always_allow_html: true
# pdf_document: default
# html_document:
# word_document: default
# # bookdown::pdf_document2: default
# bookdown::html_document2: default
# df_print: paged
# number_sections: true
# bookdown::word_document2: default
bibliography: merged.bib
csl: apa.csl
---
## Authors:
- David Janků
[1] Centre for Science, Technology, and Society Studies, Institute of Philosophy, Czech Academy of Sciences, Czech Republic; [2] Institute of Sociological Studies, Faculty of Social Sciences, Charles University, Czech Republic;
- Radim Hladík
[1] Centre for Science, Technology, and Society Studies, Institute of Philosophy, Czech Academy of Sciences, Czech Republic
## Acknowledgements
This work was supported by the Czech Science Foundation (project no. GJ20-01752Y, Funded and Unfunded Research in the Czech Republic: Scientometric Analysis and Topic Modeling)
## Declaration of Interest
☐ The authors declare that they have no known competing financial interests or personal relationships that could have appeared to influence the work reported in this paper.
☒ The authors declare the following financial interests/personal relationships which may be considered as potential competing interests:
David Janku reports a relationship with Czech Science Foundation that includes: employment. If there are other authors, they declare that they have no known competing financial interests or personal relationships that could have appeared to influence the work reported in this paper.
## Corresponding Author
Name: David Janků
Affiliation:
[1] Centre for Science, Technology, and Society Studies, Institute of Philosophy, Czech Academy of Sciences, Czech Republic
[2] Institute of Sociological Studies, Faculty of Social Sciences, Charles University, Czech Republic
Address: U Kříže 661/8, Praha 5 – Jinonice
E-mail: [email protected]
\newpage
---
```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE, warning=FALSE)
library(targets)
library(tidyr)
library(dplyr)
library(cobalt)
library(knitr)
library(psych)
library(kableExtra)
library(ggplot2)
library(lme4)
library(lmerTest)
library(MuMIn)
library(ez)
library(plm)
library(effectsize)
library(nlme)
```
# Abstract
Independence is an important factor in research excellence, often assessed for hiring, tenure, and promotion in academia. From a macro perspective, increased independence might also lead to faster exploration of the cognitive map of science and thus accelerate innovation. Many national research funders have created programmes specifically aimed at increasing the independence of early career researchers (ECRs). A quantifiable measure of such programmes' impact on changes in independence has long been lacking. We employed a recently proposed independence indicator [@vandenbesselaarMeasuringResearcherIndependence2019], to assess whether one such funding programme operated by Czech Science Foundation succeeds at stimulating independence. The indicator tracks overlaps in collaboration networks and topic focus between ECRs and their Ph.D. advisors. We found that the funding scheme designed for ECRs is not effective at fostering their independence. Regardless of the type of funding, the strongest effect on independence growth is time, explaining around 60% of the variance in independence. Additionally, we found differences in levels of independence across disciplines that were stable across time and treatment groups. Researchers in Social Sciences and Humanities demonstrated the highest independence scores, while those in Agricultural and Veterinary Sciences, and Medical and Health Sciences, showed the lowest. However, observed changes in independence across time and disciplines were caused by changes in collaboration patterns, while there were no changes in cognitive patterns across disciplines and only modest changes across time (from 40 % to 46 % of topics not shared by their PhD advisor).
## Keywords
research independence; early career researchers; project funding; mentoring; counter-factual analysis; research evaluation
# Introduction
The development of an independent research agenda is important not only to the career prospects of early career researchers (ECRs) but contributes to the advancement of science as a whole. Independent ECRs help accelerate scientific progress and push state of the art beyond the status quo. They are more likely to come up with new topics and ideas [@packalen2019] and challenge commonly held views [@vandenbesselaarMeasuringResearcherIndependence2019, p.2].
The need to cultivate research independence is a recognized objective in science policies, leading to the creation of grant programmes targeting Early Career Researchers (ECRs) by governments and funding agencies. Yet, the effectiveness of these financial incentives in fostering independence has not been empirically assessed.
An independent investigator is defined by the US National Academies of Science [@nationalresearchcouncilus2005] as “one who enjoys the independence of thought—the freedom to define the problem of interest and to choose or develop the best strategies and approaches to addressing that problem–especially during the earlier career phase” [@vandenbesselaarMeasuringResearcherIndependence2019, p.4]. Such a definition does not necessarily put independence in relation to other actors (e.g. one’s PhD advisor), but only in relation to one’s own freedom and resources. However, other institutions, such as European Research Council via their Starting Grants scheme [@neufeldPeerReviewbasedSelection2013], think of independence in a relational manner, in terms of having few publications without the former PhD supervisor being co-author; being PI on a grant; or being lab director or research leader.
## Literature review
A considerable obstacle on the road to independence is the tendency of novices to mirror the research agendas of their mentors. This phenomenon is common because following established traditions is often seen as a safer and more advantageous option for ECRs, despite the significant importance of independence in hiring procedures. Partnering with well-established seniors can provide significant career benefits [@li2019]. Being thematically aligned with their PhD and post-doc advisors increases the likelihood of continuing in academia and having more mentees in the future [@lienard2018]. Empirical evidence indicates that a majority of researchers choose this strategy [@foster2015].
<!-- To what extent is scientific progress conserved and impeded by novices mirroring seniors’ research agendas? Such a phenomenon is likely to occur because following tradition is usually a more beneficial and safer option for novices. Indeed, empirical data suggest that this strategy is chosen by a majority of researchers [@foster2015], and only a small fraction dares to deviate and explore the cognitive map of science more rapidly. For early career researchers specifically, partnering with already well-established seniors provides a large career boost [@li2019] and being thematically closer to their PhD as well as their post-doc advisor increases the probability of continuing in academia and having more mentees later on [@lienard2018]. -->
However, this effect might stifle scientific progress as it leads to a slower exploration of its cognitive map [@rzhetsky2015]. Moreover, this mirroring further strengthens the existing positions of senior faculty, making little space for disrupting the status quo and creating new perspectives.
This was illustrated by @azoulay2010, who investigated the effects of unexpected deaths of prominent scientists in their research fields and their collaborators' careers. They found a decrease in the productivity of junior collaborators and an increase in published work by young newcomers in the field rewarded by an increased citation rate. That suggests the sudden departures of prominent scientists gave room for more disruptive changes in the field and for bringing new ideas and perspectives while harming the status quo holders who worked with the prominent scientists on their research agendas - a “Goliath effect,” as coined by @wang2021, p.42.
This mirroring effect is also closely connected to the concept of academic inbreeding [@machacek2022], which focuses more on academic mobility and conservation of institutional habitus rather than specifically on the inheritance of research agendas. However, inbreeding is also often thought to increase dependence and decrease exploration of new ideas.
<!-- As younger researchers are already more likely to come up with new topics and ideas [@packalen2019], it would be fitting to empower them and stimulate their independence so that they can “challenge the existing common views and open up new lines of research” [@vandenbesselaarMeasuringResearcherIndependence2019, p.2] and accelerate scientific progress rather than recreate the existing status quo. -->
<!-- An independent investigator is then defined by the US National Academies of Science [@nationalresearchcouncilus2005] as “one who enjoys the independence of thought—the freedom to define the problem of interest and to choose or develop the best strategies and approaches to addressing that problem–especially during the earlier career phase” [@vandenbesselaarMeasuringResearcherIndependence2019, p.4]. Such a definition does not necessarily put independence in relation to other actors (e.g. one’s PhD advisor), but only in relation to one’s own freedom and resources. However other institutions, such as European Research Council via their Starting Grants scheme [@neufeldPeerReviewbasedSelection2013], think of independence in a relational manner, in terms of having one (or a few) publications without the former PhD supervisor being co-author; being PI on a grant; or being lab director or research leader. -->
Developing independence is also one of the things that the system of science expects from its members - it is one of the criteria often used for assessing researchers for hiring, tenure and promotion [@vanarensbergenDifferentViewsScholarly2014; @moherAssessingScientistsHiring2018], constituting one factor of research excellence. Even from the perspective of scholarly output, protégés were found to “do their best work when they break from their mentor’s research topics and coauthor no more than a small portion of their overall research with their mentors” [@ma2020]. @lienard2018 found that making a larger thematic step from the PhD advisor and finding a post-doc advisor with less similar interests was also related to increased chances of continuing in academia and having more mentees later on (although under the condition that one remains thematically close to the post-doc advisor). There is also some direct preliminary evidence that higher independence modestly correlates with the number of citations acquired [@rojkoScientificPerformanceResearch2022], but others argue that even the research systems with very low independence can create a lot of scientific impact, and thus the relationships between independence and impact might not be as clear [@glaserIndependenceResearchReview2022].
However, developing this independence does not go without saying. There are few known factors influencing the level of research independence. One of them is study discipline. Based on some early empirical studies [@rojkoScientificPerformanceResearch2022], it seems that researchers in social sciences and humanities experience significantly higher levels of independence from their PhD advisor than researchers in other disciplines.
Interestingly, there is also some evidence that the propensity to cognitive independence might be inherited and passed down from generation to generation in mentor-mentee relationships [@yoshioka-kobayashi2021] - perhaps it is not only research agendas that students inherit from their advisors but also the tendency to deviate from those research agendas.
In recent decades, there has been a worry that researchers’ independence is decreasing, illustrated by an increase in the age at which researchers received their first grant [@powell2016; @christian2021; and @nationalresearchcouncilus2005; @danielsGenerationRiskYoung2015 for US biomedical researchers specifically], as well as direct measures of new researchers becoming more dependent on their PhD advisors across almost all disciplines in Slovenia [@rojkoScientificPerformanceResearch2022, table 6 on p.9]. These trends have motivated some to create new proposals for funding agencies and other research funders in order to support early career researchers [@dewinde2021]
Perhaps to tackle this trend, many grant agencies decided to support early career researcher by funding. Some have decided to address this trend by giving advantage to first-awardees and early career researchers in their usual distribution process, while others have created specific programmes dedicated to funding young researchers, typically up to 8 years after completing their PhD. Such programmes have often explicitly stated goals of helping researchers become more independent, set up their own research groups and open up new lines of research. Examples of such programmes could be Starting Grants by European Research Council [@StartingGrantERC], Junior project and Junior Star by @czechsciencefoundation2019, Walter N. Benjamin Programme by DFG [@WalterBenjaminProgramme], Innovation Research Incentives Scheme by Netherlands Organization of Scientific Research [@nwo2023].
Grant funding mechanisms have previously been shown to stimulate productivity in terms of the number of publications and impact in terms of citations [@gallo2018; for the Czech environment specifically see @bajgar2021; notable exception is @vandenbesselaar2015, who specifically looked at funding programmes for early career researchers and found small or no effect], and even scientific novelty [@wang2018, or, as some authors argue, rather interdisciplinarity: @fontana2020]. In terms of career progress, obtaining grant funding was connected to significantly higher chances of career progression and obtaining a professorship [@bloch2014, @vandenbesselaar2015].
Funding is thus thought to be one of the mechanisms that could stimulate researchers' independence. However, there hasn’t been much effort focused on quantifying this phenomenon. Some studies try to measure independence in a simple way, as a ratio of publications the given researcher produced without their supervisor divided by the total number of publications of the given researchers [e.g. @rojkoScientificPerformanceResearch2022]. However, such an indicator could be misleading in the world of team science, and also only focuses on collaborations, not tracking whether a given researcher becomes independent in their thinking and research topics. @vandenbesselaarMeasuringResearcherIndependence2019 therefore proposed a more complex indicator for measuring researcher independence on the individual level, based on co-authorship network and topical overlaps. Their measure uses only information about publications’ characteristics (authorship and reference list) regardless of their impacts, unlike other alternative measures that focus solely on the impact of the independent work in terms of received citations [@dey2023].
Understanding the impact of early-career funding on research independence is crucial for shaping future funding policies and, ultimately, the trajectory of scientific innovation. This study serves as a pivotal step in quantifying that impact, providing actionable insights for funding agencies and academic institutions alike.
# Method
## Sample / data
We used the database of all publications produced by researchers employed at Czech research institutions (RIV) combined with a database of all publicly funded research projects and grants in the Czech Republic (CEP), both publicly accessible at https://www.isvavai.cz/, and transformed into a single database with cleaned data, unique person and publication identifiers, and better structure (Hladik, XXX).
This allowed us to identify all researchers who received Junior Grants from the Czech Science Foundation, a funding programme that has an explicit goal to help early career researchers become independent and set up their own research groups. There were 491 projects funded from 2015 to 2020, usually 3 years long (with some being only 2 years long), where the PI must have been up to 8 years after their PhD defence (with potential extension of 2 years for each child the researcher had in that period) and the grant could contribute up to 100 % of their salary (usually, this is limited to 50 % in other funding programmes). From these 491 funded projects, we selected 348 (71 %) grants of the recipients from Czechia, since the remaining recipients were from abroad, and thus would not have their previous publications recorded in the database of researchers employed at Czech research institutions (RIV) that we used for all analyses. There were also 7 cases where one researcher received this grant two times - in these cases, we counted only the first project that was funded, reducing the number of recipients we worked with to 341.
In the second step, we manually searched for PhD advisors of these 341 researchers in the databases of theses and dissertations that cover most universities in Czechia (https://theses.cz and https://dspace.cuni.cz/). We were able to match `r nrow(tar_read(ids_complete) %>% filter(!is.na(sup_name)))` researchers with a PhD advisor (82 % of the Czech subsample) and find suitable matched pair from control group for each of them.
<!-- Further, we have not managed to find a suitable matched pair for another `r nrow(tar_read(ids_complete))-nrow(tar_read("matching") %>% filter(treatment == 1) %>% distinct(vedidk))` researchers in the matching procedure, reducing our final sample to `r nrow(tar_read("matching") %>% filter(treatment == 1) %>% distinct(vedidk))`. -->
For the matched researchers from control groups see the matching description below in the Analysis section.
These researchers and their PhD advisors with their full publication history (limited to publications created at Czech institutions) constituted our final sample.
We also tested an assumption that had a hope of reducing time intensity of data collection by substituting it with an algorithmic approach: using the "most frequent co-author of the first 5 publications" as a proxy to identify supervisors. Crosschecking this proxy with the actual data about supervisors we found manually showed that this algorithm was able to identify supervisors correctly in only about 56 % of cases (see the Attachments).
## Measures
*Independence*
We decided to use the recently created independence indicator by @vandenbesselaarMeasuringResearcherIndependence2019, since it is based on scientific production (rather than impact) and captures the phenomena in the most complete manner. The indicator has 4 parts: I1: The eigenvector centrality of the supervisor in the researcher’s co-author network; I2: The clustering coefficient of the supervisor in the researcher’s co-author network; I3: The share of own papers of the researcher Pns/Pall where Pns is the number of publications of the researcher, not coauthored with the former supervisor(s) and Pall is the total number of publications of the researcher; and I4: The share of own research topics of the researcher Tns/Tall where Tns is the number of research topics of the researcher in which the former supervisor(s) is not active, and Tall is the total number of topics of the researcher.
However, we have made some changes to the indicator. After consulting with its authors, we have changed the aggregation formula to
$$RII = ((1- I1) + I2 + I3 + I4) / 4$$
since the previous formula had a small error (it weighted I4 double,, i.e. 2*I4). In determining the proportion of a researcher's own research topics (I4), we opted for text-based topic modeling utilizing publication titles and abstracts, diverging from the originally suggested bibliographic coupling method. Initially, we conducted a random sampling of 3000 publications from a set of all publications produced by researchers affiliated with Czech institutions as cataloged in the RIV database, estimated number of topics using Griffiths2004 indicator, and calculated LDA topic model on this set of publications. Then we compiled all publications generated by the researchers within our sample and their respective supervisors. We then calculated the posterior distribution of topics for this compilation, employing a pre-trained topic model. This way we wanted to mitigate the endogeneity of the topic modelling which would be stronger if we calculated topic model directly on the publications of our sample researchers.
To calculate this indicator, we required at least 3 publications. This led to an increased number of missing observations (see section Descriptives - Missing data).
<!-- Finally, the original indicator, while tracking not only collaboration independence but also topical independence, is still skewed significantly towards collaboration independence (three-quarters of the indicator measure collaboration, while only one-quarter measures topics). Since we consider cognitive/topical independence an important asset of this indicator, we have decided to adjust the formula a bit further and drop the I3 part, to give more weight to the cognitive/topical independence in the whole indicator. Throughout this paper, we will thus refer to the original indicator as RIIo, our adjusted version as RIIa and the part of the indicator measuring only the cognitive independence RIIc. -->
*Career age*
We measured researchers’ career age as the number of years since their first publication.
*Discipline*
We used the highest level of the OECD classification of disciplines: containing categories Natural Sciences; Engineering and Technology; Medical and Health Sciences; Agricultural Sciences; Social Sciences; Humanities. We chose this classification because it is widely used and also present in the dataset we worked with. For the matching we used the 42 sub-categories of this classification to achieve more granularity.
We attributed a discipline to a given researcher based on disciplines attributed to their each publication, choosing the most frequent discipline for each author.
*Interdisciplinary proportion*
This measure was calculated as the ratio of publications authored by the given researcher which were not assigned to the author’s main discipline.
*Number of publications*
We counted all publications the given researcher produced before the intervention year (i.e. receiving a Junior Grant) while affiliated with the Czech institution (our database only contained publications affiliated with Czech institutions). We only counted publications of types: journal articles, conference proceedings, book chapters, and full books.
*Number of publications in Web of Science and Scopus*
We counted all publications the given researcher produced before the intervention year (i.e. receiving a Junior Grant) while affiliated with the Czech institution (our database only contained publications affiliated with Czech institutions), that were also linked in Web of Science or Scopus (which means they were likely published in more recognised journals). We only counted publications of types: journal articles, conference proceedings, book chapters, and full books.
*Number of grants received prior to Junior Grant*
We calculated how many (Czech) grants each researcher has received prior and 3 years after the intervention year (i.e. receiving a Junior Grant). We did not use this variable for matching, but for filtering out control group researchers pre-matching.
*Career age when receiving the first grant*
We calculated the career age of each researcher when they received their first grant.
*Gender*
We used Genderize.io API to estimate the probability of perceived gender based on the first names of researchers. We had also manually specify gender for 2 observations from the treatment group for which the gender was not calculated automatically.
## Analysis
We analysed the data in the R version 4.2.3. using R studio and [this code](https://github.com/david-janku/juniors). For the construction of the independence indicator, we used network modelling and text-based topic modelling.
For the construction of the control group, we implemented propensity matching using the package MatchIt [@ho2011] and matching exactly by the discipline (42 sub-categories of OECD classification) and treatment year, and approximately by number of publications in total, number of publications in WoS/Scopus, career age, interdisciplinary proportion, number of grants received prior to Junior Grant, career age when receiving the first grant, and gender. All of these variables were calculated in the intervention year (i.e. year when the given researcher from experimental group received the Junior Grant). The distance was set to "mahalanobis". The matching was done in 1:1 ratio and with replacement.
Using the matching algorithm, we have created 2 control groups: "Unfunded" group did not receive any grants prior to and 3 years after the treatment year. "Funded" group was matched based on career year at which they received their first grant, ensuring that while treatment group researchers received Junior Grant, "funded" researchers received a different grant at the same time. That allowed us to better track the impact of any grant funding vs specifically Junior Grants funding on researchers’ independence.
The difference between treatment groups was tested using repeated measures ANOVA with a special error term taking into account paired nature of the data. The ANOVA model was specified as RII ~ treatment * independence_timing, incorporating both the main effects and the interaction between treatment and independence timing. A significant aspect of our ANOVA approach was the inclusion of an error term structured as Error(subclass/independence_timing). This error structure accounts for the nested or hierarchical nature of our data, where measurements are nested within subclasses and vary across different independence timings. By doing so, the model adjusts for potential within-subclass correlations and differences across independence timings.
To ascertain the significance of academic field as a predictor, we constructed two additional GLS models. The first model included treatment, independence timing, and their interaction as predictors (treatment + independence_timing + treatment:independence_timing), while the second model added the main effect of the academic field (+ field). Both models employed a compound symmetry correlation structure (corCompSymm) based on subclass and independence timing, suitable for data with equal correlation between observations. By comparing these models, we aimed to evaluate the impact of including the academic field as a predictor on the model's explanatory power, thereby testing its statistical significance in explaining variations in RII.
## Descriptives
### Missing data
```{r echo=FALSE}
d <- tar_read(full_indicator)
d$disc_ford <- as.factor(d$disc_ford)
na_counts <- sum(is.na(d$RII))
na_percentage <- round((na_counts / nrow(d))*100, 1)
na_sup <- round((sum(is.na(d$sup_name))/na_counts)*100, 1)
na_pubs <- round((nrow(d %>% filter(!is.na(sup_name)) %>% filter(is.na(pubs_number)))/na_counts)*100, 1)
na_pubs_after <- round((nrow(d %>% filter(!is.na(sup_name)) %>% filter(is.na(pubs_number)) %>% filter(independence_timing == "after_intervention"))/nrow(d %>% filter(!is.na(sup_name)) %>% filter(is.na(pubs_number))))*100, 1)
na_treatment <- nrow(d %>% filter(is.na(RII)) %>% filter(treatment == 1))
na_control1 <- nrow(d %>% filter(is.na(RII)) %>% filter(treatment == 0))
na_control2 <- nrow(d %>% filter(is.na(RII)) %>% filter(treatment == 2))
na_counts_researchers <- round(nrow(d %>% filter(!is.na(sup_name)) %>% filter(is.na(pubs_number)) %>% filter(independence_timing == "after_intervention") %>% distinct(vedidk))/nrow(d %>% filter(independence_timing == "after_intervention") %>% distinct(vedidk))*100, 1)
mean_pubs_after <- d %>%
filter(independence_timing == "after_intervention") %>%
group_by(vedidk) %>%
summarise(pubs_number = first(pubs_number)) %>%
summarise(mean_pubs_number = mean(pubs_number, na.rm = TRUE)) %>%
pull()
# s <- d %>% filter(is.na(RII))
#
# f_after <- d %>% filter(!is.na(RII)) %>% filter(independence_timing == "after_intervention")
# f_before <- d %>% filter(!is.na(RII)) %>% filter(independence_timing == "before_intervention")
#
#
# v_after <- c(f_after$vedidk[f_after$treatment == 1 ], NA)
# v_before <- c(f_before$vedidk[f_before$treatment == 1 ], NA)
#
# r_after <- f_after %>% filter(vedidk_treatment %in% v_after)
# r_before <- f_before %>% filter(vedidk_treatment %in% v_before)
#
#
#
# v_after_control <- c(r_after$vedidk[r_after$treatment != 1 ], r_after$vedidk_treatment)
# v_before_control <- c(r_before$vedidk[r_before$treatment != 1 ], r_before$vedidk_treatment)
#
# r_after_final <- r_after %>% filter(vedidk %in% v_after_control)
# r_before_final <- r_before %>% filter(vedidk %in% v_before_control)
# r_test <- r_after %>% filter(treatment != 2)
# v_after_test <- c(r_test$vedidk[r_test$treatment != 1 ], r_test$vedidk_treatment)
# r_test_final <- r_test %>% filter(vedidk %in% v_after_test)
# final_before_control1 <- nrow(r_before_final %>% filter(treatment == 0))
# final_before_control2 <- nrow(r_before_final %>% filter(treatment == 2))
#
#
# final_after_control1 <- nrow(r_after_final %>% filter(treatment == 0))
# final_after_control2 <- nrow(r_after_final %>% filter(treatment == 2))
# a <- intersect(
# r_after_final$vedidk,
# r_before_final$vedidk
# )
#
#
# all <- rbind(r_before_final, r_after_final) %>% filter(vedidk %in% a)
filtered_by_subclass <- d %>%
group_by(subclass, independence_timing) %>%
filter(!any(is.na(RII))) %>%
ungroup()
filtered_subclass <- nrow(filtered_by_subclass %>% distinct(subclass))
subclass_all <- nrow(d %>% distinct(subclass))
filtered_subclass_perc <- round((filtered_subclass/subclass_all)*100, 1)
final_before_control1 <- nrow(filtered_by_subclass %>% filter(independence_timing == "before_intervention") %>% filter(treatment == 0))
final_before_control2 <- nrow(filtered_by_subclass %>% filter(independence_timing == "before_intervention") %>% filter(treatment == 2))
final_after_control1 <- nrow(filtered_by_subclass %>% filter(independence_timing == "after_intervention") %>% filter(treatment == 0))
final_after_control2 <- nrow(filtered_by_subclass %>% filter(independence_timing == "after_intervention") %>% filter(treatment == 2))
```
The independence indicator was not possible to calculate for `r na_counts` (`r na_percentage` % of observations). This is in `r na_sup` % caused by missing data about supervisors, and in `r na_pubs` % due to absence of publications by the researcher during the specified period (`r na_pubs_after` % of these cases occurred post-intervention, often in the past few years).
There are `r na_treatment` missing observations in the treatment group, `r na_control1` in the unfunded control group and `r na_control2` in the funded control group.
If we only keep observations that have a match with non-missing values, we are left with a sample of `r final_before_control1` matched observations in unfunded control group and `r final_before_control2` matched observations in funded control group before intervention and a sample of `r final_after_control1` matched observations in unfunded control group and `r final_after_control2` matched observations in funded control group after intervention.
From a perspective of disciplines, we see that there are slightly more missing data in Agricultural and veterinary sciences and the Humanities and the Arts.
```{r echo=FALSE}
dt <- d
dt$missing <- ifelse(is.na(dt$RII), "Missing", "Complete")
# Calculate counts and proportions
df_summary <- dt %>%
group_by(field, missing) %>%
summarise(count = n(), .groups = "drop") %>%
mutate(Proportion = count / sum(count)) %>%
pivot_wider(names_from = missing, values_from = c(count, Proportion))
# If there are any NA values in the 'count_Missing' column, replace them with 0
df_summary$count_Missing[is.na(df_summary$count_Missing)] <- 0
# Calculate Missing / (Missing + Complete) in percentages for each disc_ford
df_summary <- df_summary %>%
mutate(Proportion = ifelse(!is.na(`count_Missing`) & !is.na(`count_Complete`), `count_Missing` / (`count_Missing` + `count_Complete`) * 100, NA))
# If there are any NA values in the 'Proportion' column, replace them with 0
df_summary$Proportion[is.na(df_summary$Proportion)] <- 0
# Remove unnecessary columns and rename the others
df_summary <- df_summary %>%
select(field, count_Complete, count_Missing, Proportion) %>%
rename(
Field = field,
Complete = count_Complete,
Missing = count_Missing,
Proportion_Missing = Proportion
)
# Print the updated table
kable(df_summary, format = "markdown", digits = 3, caption = "Counts and Proportions of Missing Data by Field")
```
### Before intervention
```{r echo=FALSE}
df <- filtered_by_subclass %>% filter(independence_timing == "before_intervention")
numeric_vars <- df %>%
select(where(is.numeric)) %>%
select(-weights, -funded_control, -id, -treatment_year, -career_start_year) %>%
names()
# descriptive stats for each treatment level
table_before_1 <- df %>% filter(treatment == 1) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
table_before_0 <- df %>% filter(treatment == 0) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
table_before_2 <- df %>% filter(treatment == 2) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
colnames(table_before_0) <- paste0(colnames(table_before_0), "_0")
colnames(table_before_2) <- paste0(colnames(table_before_2), "_2")
combined_table_before <- cbind(table_before_1, table_before_0, table_before_2)
kable(combined_table_before,
format = "markdown", digits = 3, align = "c", booktabs = TRUE, caption = "Descriptives of All Groups Before Intervention")
# kableExtra::add_header_above(c(" " = 1, "Treatment Group" = 5, "Unfunded Control Group" = 5, "Funded Control Group" = 5))
#### below is html version - comment it away if you are printing in doc
# df <- filtered_by_subclass %>% filter(independence_timing == "before_intervention")
# numeric_vars <- df %>%
# select(where(is.numeric)) %>%
# select(-weights, -funded_control, -id, -treatment_year, -career_start_year) %>%
# names()
#
# # descriptive stats for each treatment level
# table_before_1 <- df %>% filter(treatment == 1) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
# table_before_0 <- df %>% filter(treatment == 0) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
# table_before_2 <- df %>% filter(treatment == 2) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
#
# colnames(table_before_0) <- paste0(colnames(table_before_0), "_0")
# colnames(table_before_2) <- paste0(colnames(table_before_2), "_2")
#
# combined_table_before <- cbind(table_before_1, table_before_0, table_before_2)
#
# kable(combined_table_before,
# digits = 3, align = "c", booktabs = TRUE, caption = "Descriptives of All Groups Before Intervention") %>%
# kableExtra::kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive")) %>%
# add_header_above(c(" " = 1, "Treatment Group" = 5, "Unfunded Control Group" = 5, "Funded Control Group" = 5))
```
### After intervention
```{r echo=FALSE}
df <- filtered_by_subclass %>% filter(independence_timing == "after_intervention")
df$career_length2022 <- 2022-df$career_start_year
numeric_vars <- df %>%
select(where(is.numeric)) %>%
select(-weights, -funded_control, -id, -treatment_year, -career_start_year, -career_length, -pubs_total, -ws_pubs, -interdisc_proportion, -grants, -total_coauthor_count, -first_grant) %>%
names()
# descriptive stats for each treatment level
table_after_1 <- df %>% filter(treatment == 1) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
table_after_0 <- df %>% filter(treatment == 0) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
table_after_2 <- df %>% filter(treatment == 2) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
colnames(table_after_0) <- paste0(colnames(table_after_0), "_0")
colnames(table_after_2) <- paste0(colnames(table_after_2), "_2")
# Move 'career_length2022' row to the top for each table
reorder_row <- function(tbl) {
return(tbl[c("career_length2022", setdiff(rownames(tbl), "career_length2022")), ])
}
table_after_1 <- reorder_row(table_after_1)
table_after_0 <- reorder_row(table_after_0)
table_after_2 <- reorder_row(table_after_2)
# Re-combine tables
combined_table_after <- cbind(table_after_1, table_after_0, table_after_2)
kable(combined_table_after,
format = "markdown", digits = 3, align = "c", booktabs = TRUE, caption = "Descriptives of All Groups After Intervention") %>%
kableExtra::add_header_above(c(" " = 1, "Treatment Group" = 5, "Unfunded Control Group" = 5, "Funded Control Group" = 5))
# Assuming table_after_1, table_after_0, table_after_2 are your data frames for different groups
#
# # Assuming each data frame has 5 columns
# header_columns = c("Header", rep("", 4))
# header_1 <- data.frame(matrix(nrow = 1, ncol = 5))
# colnames(header_1) <- header_columns
# header_1[1, 1] <- "Treatment Group"
#
# header_0 <- data.frame(matrix(nrow = 1, ncol = 5))
# colnames(header_0) <- header_columns
# header_0[1, 1] <- "Unfunded Control Group"
#
# header_2 <- data.frame(matrix(nrow = 1, ncol = 5))
# colnames(header_2) <- header_columns
# header_2[1, 1] <- "Funded Control Group"
#
# header_names <- colnames(table_after_1) # Assuming all tables have the same structure
# header_row <- setNames(data.frame(matrix(ncol = length(header_names), nrow = 1), stringsAsFactors = FALSE), header_names)
# header_row[1, 1] <- "Your Header Title" # Set your header title in the first column
#
# combined_table_after <- rbind(header_row, table_after_1, header_row, table_after_0, header_row, table_after_2)
#
#
# # # Assuming table_after_1, table_after_0, table_after_2 have the same column names
# # combined_table_after <- rbind(header_1, table_after_1,
# # header_0, table_after_0,
# # header_2, table_after_2)
#
# # Now create the table in R Markdown
# knitr::kable(combined_table_after, format = "markdown", digits = 3,
# caption = "Descriptives of All Groups After Intervention")
# # Table for Treatment Group 1
# table_after_1_md <- knitr::kable(table_after_1, format = "markdown",
# caption = "Treatment Group 1")
#
# # Table for Unfunded Control Group
# table_after_0_md <- knitr::kable(table_after_0, format = "markdown",
# caption = "Unfunded Control Group")
#
# # Table for Funded Control Group
# table_after_2_md <- knitr::kable(table_after_2, format = "markdown",
# caption = "Funded Control Group")
#
#### below is html version - comment it away if you are printing in doc
# df <- filtered_by_subclass %>% filter(independence_timing == "after_intervention")
# df$career_length2022 <- 2022-df$career_start_year
# numeric_vars <- df %>%
# select(where(is.numeric)) %>%
# select(-weights, -funded_control, -id, -treatment_year, -career_start_year, -career_length, -pubs_total, -ws_pubs, -interdisc_proportion, -grants, -total_coauthor_count, -first_grant) %>%
# names()
#
# # descriptive stats for each treatment level
# table_after_1 <- df %>% filter(treatment == 1) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
# table_after_0 <- df %>% filter(treatment == 0) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
# table_after_2 <- df %>% filter(treatment == 2) %>% distinct(vedidk, .keep_all = TRUE) %>% select(all_of(numeric_vars)) %>% describe() %>% select(n, mean, sd, min, max) %>% round(., digits = 3)
#
# colnames(table_after_0) <- paste0(colnames(table_after_0), "_0")
# colnames(table_after_2) <- paste0(colnames(table_after_2), "_2")
#
# # Move 'career_length2022' row to the top for each table
# reorder_row <- function(tbl) {
# return(tbl[c("career_length2022", setdiff(rownames(tbl), "career_length2022")), ])
# }
#
# table_after_1 <- reorder_row(table_after_1)
# table_after_0 <- reorder_row(table_after_0)
# table_after_2 <- reorder_row(table_after_2)
#
# # Re-combine tables
# combined_table_after <- cbind(table_after_1, table_after_0, table_after_2)
#
# kable(combined_table_after,
# digits = 3, align = "c", booktabs = TRUE, caption = "Descriptives of All Groups After Intervention") %>%
# kableExtra::kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive")) %>%
# add_header_above(c(" " = 1, "Treatment Group" = 5, "Unfunded Control Group" = 5, "Funded Control Group" = 5))
```
## Matching
```{r echo=FALSE}
removed_treatment_unfunded <- (tar_read(ids_complete) %>% distinct(vedidk) %>% nrow())-(tar_read(matching) %>% group_by(subclass) %>% filter(!any(treatment == 2)) %>%
ungroup() %>% filter(treatment == "1") %>% distinct(vedidk) %>% nrow())
removed_treatment_funded <- (tar_read(ids_complete) %>% distinct(vedidk) %>% nrow())-(tar_read(matching) %>% group_by(subclass) %>% filter(!any(treatment == 0)) %>%
ungroup() %>% filter(treatment == "1") %>% distinct(vedidk) %>% nrow())
unique_treatment_unfunded <- d %>% group_by(subclass) %>% filter(!any(treatment == 2)) %>% ungroup() %>% filter(treatment == "1") %>% distinct(vedidk) %>% nrow()
unique_treatment_funded <- d %>% group_by(subclass) %>% filter(!any(treatment == 0)) %>%
ungroup() %>% filter(treatment == "1") %>% distinct(vedidk) %>% nrow()
unique_control_unfunded <- d %>% filter(treatment == 0) %>% distinct(vedidk) %>% nrow()
unique_control_funded <- d %>% filter(treatment == 2) %>% distinct(vedidk) %>% nrow()
replaced_control_unfunded <- unique_treatment_unfunded-unique_control_unfunded
replaced_control_funded <- unique_treatment_funded-unique_control_funded
overlap_control <- unique_control_unfunded+unique_control_funded-(d %>% filter(treatment != 1) %>% distinct(vedidk) %>% nrow())
```
After matching, all standardized mean differences for the covariates were below 0.1 and all standardized mean differences for squares and two-way interactions between covariates were below .15, indicating adequate balance.
Matching process removed `r removed_treatment_unfunded` treatment observations from matching with unfunded control group and `r removed_treatment_funded` treatment observations from matching with funded control group. This observation was removed because the researchers had no publications prior to the year of application for the grant, which would make it not possible to calculate the independence score for them anyway.
The resulting sample had `r unique_treatment_unfunded` unique treatment group observations paired with `r unique_control_unfunded` unique unfunded control group observations suggesting that `r replaced_control_unfunded` observations from unfunded control group were matched to multiple treatment group observations. Similarly, there were `r unique_treatment_funded` unique treatment group observations paired with `r unique_control_funded` unique funded control group observations, suggesting that `r replaced_control_funded` funded control group observations were matched to multiple treatment group observations.
There is an overlap between control group observations: `r overlap_control` observations from the unfunded control group also appears in the funded control group, showing that they did eventually received grant at the similar time as those from treatment group.
```{r echo=FALSE, fig.cap="Covariate balance table for the unfunded control group."}
# # Create a subset for treatment group and control group 1
# subset_1 <- (tar_read(matching)) %>% filter(treatment != 2)
# subset_2 <- (tar_read(matching)) %>% filter(treatment != 0)
#
# # Generate balance table for treatment vs control group 1
# balance_results_1 <- bal.tab(treatment ~ career_lenght + pubs_total + ws_pubs + interdisc_proportion + grants + gender, data = subset_1, disp = c("mean", "std"))
# balance_results_2 <- bal.tab(treatment ~ career_lenght + pubs_total + ws_pubs + interdisc_proportion + grants + gender, data = subset_2, disp = c("mean", "std"))
#
# # Define a function to manually extract balance statistics from bal.tab object
# extract_balance_stats <- function(bal_results) {
# covariates <- rownames(bal_results$Balance)
# means_treat <- bal_results$Balance$`M.1.Un`
# means_ctrl <- bal_results$Balance$`M.0.Un`
# diff <- bal_results$Balance$`Diff.Un`
#
# data.frame(Covariate = covariates,
# Mean_Treatment = means_treat,
# Mean_Control = means_ctrl,
# Diff = diff)
# }
#
# # Extract balance stats for subset_1 and subset_2
# balance_stats_1 <- extract_balance_stats(balance_results_1)
# balance_stats_2 <- extract_balance_stats(balance_results_2)
#
# # Print tables in R Markdown
# knitr::kable(balance_stats_1, caption = "Balance Statistics: Treatment vs Control Group 1") %>% kableExtra::kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive"))
# knitr::kable(balance_stats_2, caption = "Balance Statistics: Treatment vs Control Group 2") %>% kableExtra::kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive"))
##Unfunded group
unf_sum <- summary(tar_read(matched_obj_unfunded), un = TRUE)
# summary(tar_read(matched_obj_unfunded))
#
# plot(unf_sum, var.order = "unmatched")
aa <- cobalt::bal.tab(tar_read(matched_obj_unfunded), un = TRUE, stats = c("m", "v", "ks"), binary = "std")
# aa <- aa[!grepl("^disc_ford", names(aa))]
# aa <- aa[!grepl("^disc_ford", names(aa$Balance))]
bal_df <- aa$Balance
bal_df <- bal_df[!grepl("^disc_ford", rownames(bal_df)), ]
# Display the cleaned-up table with kable
kable(bal_df, format = "markdown", caption = "Summary of Covariate Balance Post-Matching For Unfunded Control Group", digits = 3)
aa$Balance <- bal_df
cobalt::love.plot(aa, binary = "std", var.order = "adjusted")
```
```{r echo=FALSE, fig.cap="Covariate balance table for the funded control group."}
##Funded group
fun_sum <- summary(tar_read(matched_obj_funded), un = TRUE)
# summary(tar_read(matched_obj_funded))
#
# plot(fun_sum, var.order = "unmatched")
bb <- cobalt::bal.tab(tar_read(matched_obj_funded), un = TRUE, stats = c("m", "v", "ks"), binary = "std")
# aa <- aa[!grepl("^disc_ford", names(aa))]
# aa <- aa[!grepl("^disc_ford", names(aa$Balance))]
bal_df_funded <- bb$Balance
bal_df_funded <- bal_df_funded[!grepl("^disc_ford", rownames(bal_df_funded)), ]
# Display the cleaned-up table with kable
kable(bal_df_funded, format = "markdown", caption = "Summary of Covariate Balance Post-Matching For Funded Control Group", digits = 3)
bb$Balance <- bal_df_funded
cobalt::love.plot(bb, binary = "std", var.order = "adjusted")
```
# Results
## Main results
To ascertain the differences in independence scores before and after intervention, while accounting for the treatment type, academic disciplines, and paired nature of the data, we used repeated measures ANOVA. This allowed us to measure the main effect of change in independence across time, across disciplines, and the main treatment effect was the interaction between treatment type and time.
<!-- POZN: only for comparison: -->
<!-- To estimate the treatment effect and its standard error, we fit a linear regression model with 1978 earnings as the outcome and the treatment, covariates, and their interaction as predictors and included the full matching weights in the estimation. The lm() function was used to fit the outcome, and the comparisons() function in the marginaleffects package was used to perform g-computation in the matched sample to estimate the ATT. A cluster-robust variance was used to estimate its standard error with matching stratum membership as the clustering variable. -->
<!-- The estimated effect was XXXX (SE = XXXX, p = XXXX), indicating that the average effect of the treatment for those who received it is to increase earnings. -->
```{r echo=FALSE}
pp <- d %>%
group_by(subclass) %>%
filter(!any(is.na(RII))) %>%
ungroup()
pp <- pp %>%
group_by(vedidk, subclass) %>% # Assuming 'id' is your current unique identifier
mutate(subject_id = cur_group_id()) %>% # This creates a unique ID for each group
ungroup()
diff_data <- pp %>%
select(subject_id, independence_timing, RII) %>%
spread(independence_timing, RII) %>%
mutate(RII_diff = after_intervention - before_intervention) %>%
select(subject_id, RII_diff)
pp <- left_join(pp, diff_data, by = "subject_id")
filtered_by_subclass_1 <- pp %>%
group_by(subclass) %>%
filter(!any(treatment == 2)) %>%
ungroup() %>%
mutate(treatment = factor(treatment, levels = c(0, 1)))
filtered_by_subclass_1$subclass <- as.factor(filtered_by_subclass_1$subclass)
filtered_by_subclass_1$treatment <- as.factor(filtered_by_subclass_1$treatment)
filtered_by_subclass_1$independence_timing <- as.factor(filtered_by_subclass_1$independence_timing)
filtered_by_subclass_1$field <- as.factor(filtered_by_subclass_1$field)
# this DiD model unfortunately doesnt work for some reason that I cannot resolve, but it is possible to calculate the same using repeated measures anova or mixes effects model, so it should be fine
# # Run regression model with interaction term
# did_model <- plm::plm(RII ~ independence_timing * treatment + factor(subclass),
# data = filtered_by_subclass_1,
# index = c("subclass"),
# model = "within")
#
# # Calculate clustered standard errors
# clustered_se <- coeftest(did_model, vcov = function(x) vcovHC(x, type = "HC1", cluster = "subclass"))
## here should be some assumptions tests that I will run to see whtehr it is correct, but later I will comment it so that it doesnt show in the final paper
# model <- lmer(RII ~ treatment * independence_timing + (1 | subclass), data = filtered_by_subclass_1)
# # summary(model)
# # r.squaredGLMM(model)
#
# model_no_interaction <- lmer(RII ~ treatment + independence_timing + (1 | subclass), data = filtered_by_subclass_1)
#
# RSS_no_interaction <- sum(residuals(model_no_interaction)^2)
# RSS_full <- sum(residuals(model)^2)
# delta_RSS <- RSS_no_interaction - RSS_full
# R2_interaction <- delta_RSS / RSS_no_interaction
# # print(R2_interaction)
#
#
# model_disc <- lmer(RII ~ treatment * independence_timing * field +
# (1 + independence_timing|subclass),
# data = filtered_by_subclass_1)
# summary(model_disc)
##checking assumptions fro repeated measures ANOVA
result <- aov(RII ~ treatment * independence_timing + Error(subclass/independence_timing), data=filtered_by_subclass_1)
# summary(result)
eta_sq_result <- eta_squared(result)
# Normality: You can check the normality of the residuals using a Q-Q plot and the Shapiro-Wilk test.
# Check residuals from lm
# plot(resid(result))
#Linearity and Homoscedasticity (Equal Variances) of Residuals: If the assumptions hold, you should not see any obvious patterns or funnel shapes in the plot.
# plot(result$fitted.values, resid(result),
# xlab = "Fitted values", ylab = "Residuals",
# main = "Residuals vs. Fitted Values")
# abline(h = 0, col = "red")
# Normality of Residuals:
# qqnorm(resid(result))
# qqline(resid(result))
#
# shapiro.test(resid(result))
# Independence of Residuals: If your data has a time sequence (e.g., repeated measurements over time), then you would be concerned about the independence of residuals. For this, you can use the Durbin-Watson test from the car package. The Durbin-Watson test statistic ranges from 0 to 4. A value around 2 suggests no autocorrelation, while values below 1 and above 3 are cause for concern.
# library(car)
# durbinWatsonTest(result)
# Multicollinearity: Check the variance inflation factor (VIF) for each predictor. Typically, a VIF above 5-10 indicates multicollinearity.
# vif(result)
# Outliers and Influence Points: Leverage vs. studentized residuals plot can be used to identify influential points.
# plot(hatvalues(result), rstudent(result),
# xlab = "Leverage", ylab = "Studentized Residuals",
# main = "Residuals vs. Leverage")
# abline(h = c(-2,2), col = "red", lty = 2)
# Sphericity: Use the mauchly.test function on your model. If the test is significant, then the assumption of sphericity has been violated.
# mauchly.test(model_aov)
# Homogeneity of Variance: Use Levene's Test. The car package offers leveneTest.
# library(car)
# leveneTest(RII ~ treatment, data = filtered_by_subclass_1)
# Outliers:You can inspect boxplots or use Mahalanobis distance.
# boxplot(RII ~ treatment, data = filtered_by_subclass_1)
##conducting repeated measures ANOVA
# filtered_by_subclass_1 <- filtered_by_subclass_1 %>%
# group_by(vedidk, subclass) %>% # Assuming 'id' is your current unique identifier
# mutate(subject_id = cur_group_id()) %>% # This creates a unique ID for each group
# ungroup()
# #this is preferable because it takes into account the matched nature of the data, as well as repeated measures nature of the data -> see chatGPT conversation "ANOVA `wid` parameter clarification"
# result <- aov(RII ~ treatment * independence_timing + Error(subclass/independence_timing), data=filtered_by_subclass_1)
# # summary(result)
# eta_sq_result <- eta_squared(result)
#this is less preferable because it takes into account only the repeated measure nature of the data, not the matched nature -> you can see that when you run this tests, there is warning that says "there were non-unique subclass values across the between-subjects variable (treatment). Essentially, ezANOVA detected the problem and internally "fixed" it by making subclass unique across the treatment groups, so it behaves similarly to subject_id you later created." -> this means that it effectively converts the subclass to subject_id anyway
# result <- ezANOVA(data = filtered_by_subclass_1,
# dv = RII,
# wid = subclass,
# within = .(independence_timing),
# between = treatment)
# # print(result)
## adding disciplines:
filtered_data_disc_1 <- filtered_by_subclass_1 %>%
group_by(field) %>%
filter(n() >= 8)
filtered_data_disc_1$field <- droplevels(filtered_data_disc_1$field)
# Calculate weights based on group size (you can adjust this based on other criteria if needed)
group_sizes <- table(filtered_data_disc_1$field)
max_size <- max(group_sizes)
filtered_data_disc_1$weights <- max_size / group_sizes[filtered_data_disc_1$field]
#this model addresses both unequal group sizes (via weights) and heteroscedasticity (via varIdent). Additionally, I've added a first-order autoregressive structure (corAR1) assuming you have repeated measures on the same subjects and to account for their paired nature (indicated by subclass/independence_timing)
#this model directly tests whether there is a difference in disciplines - anyway it tests each discipline agains each other, so it doesnt produce the neat result of "yes, there are some significant differences between disciplines", but more nuanced version of "there is specifically difference between this and this discipline"
model_gls_weighted <- gls(RII ~ field, data = filtered_data_disc_1,
weights = varIdent(form = ~ 1 | field),
method = "ML",
correlation = corAR1(form = ~ 1 | subject_id))
# summary(model_gls_weighted)
#this model below is the same as the above but with added predictors
model_gls_interactions_subclass <- gls(RII ~ field * treatment * independence_timing,
data = filtered_data_disc_1,
weights = varIdent(form = ~ 1 | field),
method = "ML",
correlation = corAR1(form = ~ 1 | subclass/independence_timing))
# aa <- summary(model_gls_interactions_subclass)
#one potential strategy to get the neat result of "yes, there are some significant differences between disciplines" is following: create two similar models
model_without_field <- gls(RII ~ treatment + independence_timing + treatment:independence_timing,
data = filtered_data_disc_1,
correlation = corCompSymm(form = ~ 1 | subclass/independence_timing),
method = "ML")
model_with_field_main_effect <- gls(RII ~ treatment + field + independence_timing + treatment:independence_timing,
data = filtered_data_disc_1,
correlation = corCompSymm(form = ~ 1 | subclass/independence_timing),
method = "ML")
anova_result <- anova(model_without_field, model_with_field_main_effect)
model_summary <- summary(model_with_field_main_effect)
# Your result specifically shows that the model_with_field_main_effect is significantly better at explaining the variance in your response variable compared to the model_without_field. The p-value is extremely small (1e-04), indicating a significant main effect of "field".
#
# In other words, the main effect of "field" significantly improves the model fit, suggesting that the variable "field" has a significant main effect on the response variable RII after accounting for other predictors in the model.
#this model does NOT address unequal group sizes (via weights) and heteroscedasticity
# result_disc <- ezANOVA(data = filtered_data_disc_1,
# dv = RII,
# wid = subclass,
# within = .(independence_timing),
# between = .(treatment, field))
# # print(result_disc)
##here I calculate the interaction of filed * treatment not from the full RII score, but from RR delta, ie change in RII from before to after intervetion. below the results are plotted, showing that the effect of treatment on disciplines is universal, expect for social sciences where there is almost no effect. it also shows that almost in all cases the effect of time is positive - i.e people are getting more independent over time, in some case by quite a lot (e.g. 0.2 in RII, which is 20 % oif the RII score)
# filtered_data_disc_1 <- filtered_data_disc_1 %>%
# filter(independence_timing != "after_intervention")
#
# means_table_disc_1 <- filtered_data_disc_1 %>%
# group_by(field, treatment) %>%
# summarise(mean_RII = mean(RII_diff, na.rm = TRUE))
#
# # Set up sum coding for "field"
# contrasts(filtered_data_disc_1$field) <- contr.sum(levels(filtered_data_disc_1$field))
#
#
# model_wls <- lm(RII_diff ~ field * treatment, data = filtered_data_disc_1, weights = weights)
# summary(model_wls)
#
#
# contrasts(filtered_data_disc_1$field)
#
# # Extract the variance-covariance matrix
# vcov_mat <- vcov(model_wls_sum_coded)
#
# # Coefficient for Social Sciences
# coef_SS <- -sum(coef(model_wls_sum_coded)[c("field1", "field2", "field3")])
#
# # Variance for Social Sciences
# var_SS <- sum(diag(vcov_mat[2:4, 2:4])) + 2*sum(vcov_mat[2,3:4]) + 2*sum(vcov_mat[3,4])
#
# # Standard Error for Social Sciences
# se_SS <- sqrt(var_SS)
#
# # t-value for Social Sciences
# t_value_SS <- coef_SS / se_SS
#
# # Degrees of Freedom
# df <- df.residual(model_wls_sum_coded)
#
# # p-value for Social Sciences
# p_value_SS <- 2 * pt(-abs(t_value_SS), df)
#
# coef_SS
# se_SS
# t_value_SS
# p_value_SS
#
# vcov_matrix <- vcov(model_wls_sum_coded)
#
#
# var_interaction <- vcov_matrix["treatment1", "treatment1"] +
# vcov_matrix["field1:treatment1", "field1:treatment1"] +
# vcov_matrix["field2:treatment1", "field2:treatment1"] +
# vcov_matrix["field3:treatment1", "field3:treatment1"] +
# 2*(vcov_matrix["treatment1", "field1:treatment1"] +
# vcov_matrix["treatment1", "field2:treatment1"] +
# vcov_matrix["treatment1", "field3:treatment1"])
#
# se_interaction <- sqrt(var_interaction)
#
# t_value <- -0.01905 / se_interaction
#
# df_residual <- df.residual(model_wls_sum_coded)
# p_value <- 2 * (1 - pt(abs(t_value), df_residual))
## visualise field*treatment
# means_table <- filtered_by_subclass_1 %>%
# filter(independence_timing != "before_intervention") %>%
# group_by(field, treatment) %>%
# summarise(mean_RII = mean(RII_diff, na.rm = TRUE))
#
# sd_counts <- filtered_by_subclass_1 %>%
# filter(independence_timing != "before_intervention") %>%
# group_by(field, treatment) %>%
# summarise(sd_RII = sd(RII_diff, na.rm = TRUE),
# n = n())
#
# means_table <- merge(means_table, sd_counts, by = c("field", "treatment")) %>%
# mutate(se_RII = sd_RII / sqrt(n))
#
#
#
#
# ggplot(means_table, aes(x=field, y=mean_RII, fill=treatment)) +
# geom_bar(stat="identity", position=position_dodge(width=0.8), width=0.7) +
# geom_errorbar(aes(ymin=mean_RII-se_RII, ymax=mean_RII+se_RII),
# position=position_dodge(width=0.8), width=0.3) +
# geom_text(aes(label=n, y= 0.5 * mean_RII), # position in the middle
# position=position_dodge(width=0.8), vjust=0.5) + # centered vertically
# theme_minimal() +
# labs(y="Mean RII Difference", x="Field", fill="Treatment", title="Comparison of differences in RII across fields and treatment group") +
# theme(axis.text.x = element_text(angle = 45, hjust = 1))