simul_these/Modules/ado/plus/m/metan7.ado

2508 lines
80 KiB
Plaintext

*! version 1.86 1Apr2004
/*Revision list at end
Syntax:
a) binary data:
metan #events_group1 #nonevents_group1 #events_group2 #nonevents_group2 , ...
b) cts data:
metan #group1 mean1 sd1 #group2 mean2 sd2 , ...
c) generic effect size+st error:
metan theta se_theta , ...
d) generic effect size+ci:
metan theta lowerlimit upperlimit , ...
*/
program define metan7 , rclass
version 7.0
#delimit ;
syntax varlist(min=2 max=6 default=none numeric) [if] [in] [, BY(string)
ILevel(integer $S_level) OLevel(integer $S_level) CC(string)
OR RR RD FIXED FIXEDI RANDOM RANDOMI PETO COHEN HEDGES GLASS noSTANDARD
CHI2 CORNFIELD LOG BRESLOW EFORM noINTeger noOVERALL noSUBGROUP SGWEIGHT
SORTBY(passthru) noKEEP noGRAPH noTABLE LABEL(string) SAVING(passthru) noBOX
XLAbel(passthru) XTick(passthru) FORCE BOXSCA(real 1.0) BOXSHA(integer 4)
TEXTS(real 1.0) T1title(string) T2title(string) LEGEND(string)
B1title(string) B2title(string) noWT noSTATS COUNTS WGT(varlist numeric max=1)
GROUP1(string) GROUP2(string) EFFECT(string) ] ;
#delimit cr
global MA_FBSH=`boxsha'
global MA_FBSC=`boxsca'
global MA_ESLA="`effect'"
if "`legend'"!="" { global S_TX="`legend'" }
else { global S_TX "Study" }
*label groups
if "`group1'"=="" { global MA_G1L "Treatment" }
else { global MA_G1L "`group1'" }
if "`group2'"=="" { global MA_G2L "Control" }
else { global MA_G2L "`group2'" }
if "`legend'"!="" { global S_TX "`legend'" }
if (`texts'>2 | `texts'<0.1 ) {local texts=1}
global MA_FTSI=`texts'
if ("`by'"=="" & "`overall'"!="") {local wt "nowt"}
if `ilevel'<1 {local ilevel=`ilevel'*100 }
if `ilevel'>99 | `ilevel'<10 { local ilevel $S_level }
global ZIND= -invnorm((100-`ilevel')/200)
if `olevel'<1 {local olevel=`olevel'*100 }
if `olevel'>99 | `olevel'<10 { local olevel $S_level }
global ZOVE= -invnorm((100-`olevel')/200)
global IND=`ilevel'
global OVE=`olevel'
#delimit ;
global S_1 "."; global S_2 "."; global S_3 "."; global S_4 ".";
global S_5 "."; global S_6 "."; global S_7 "."; global S_8 ".";
global S_9 "."; global S_10 ".";global S_11 ".";global S_12 ".";
#delimit cr
*If not using own weights set fixed as default
if "`fixed'`random'`fixedi'`randomi'`peto'"=="" & ( "`wgt'"=="" ) { local fixed "fixed" }
*declare study labels for display
if "`label'"!="" {
parse "`label'", parse("=,")
while "`1'"!="" {
cap confirm var `3'
if _rc!=0 {
di in re "Variable `3' not defined"
exit _rc
}
local `1' "`3'"
mac shift 4
}
}
tempvar code
qui {
*put name/year variables into appropriate macros
if "`namevar'"!="" {
local lbnvl : value label `namevar'
if "`lbnvl'"!="" { quietly decode `namevar', gen(`code') }
else {
gen str10 `code'=""
cap confirm string variable `namevar'
if _rc==0 { replace `code'=`namevar' }
else if _rc==7 { replace `code'=string(`namevar') }
}
}
else { gen str3 `code'=string(_n) }
if "`yearvar'"!="" {
local yearvar "`yearvar'"
cap confirm string variable `yearvar'
if _rc==7 { local str "string" }
if "`namevar'"=="" { replace `code'=`str'(`yearvar') }
else { replace `code'=`code'+" ("+`str'(`yearvar')+")" }
}
if "`wgt'"!="" {
*User defined weights verification
if "`fixed'`random'`fixedi'`randomi'`peto'"!="" {
di in re "Option invalid with user-defined weights"
exit _rc
}
confirm numeric variable `wgt'
local wgt "wgt(`wgt')"
}
} /* End of quietly loop */
parse "`varlist'", parse(" ")
if "`6'"=="" {
if "`4'"=="" {
*Input is {theta setheta} or {theta lowerci upperci} => UDW, IV or D+L weighting
if "`3'"!="" {
*input is theta lci uci
cap assert ((`3'>=`1') & (`1'>=`2'))
if _rc!=0 {
di in bl "Effect size and confidence intervals invalid:"
di in bl "order should be {effect size, lower ci limit, upper ci limit}"
exit _rc
}
}
cap assert "`log'"==""
if _rc!=0 {
di in bl "Log option not available without raw data counts: if necessary, transform both"
di in bl "effect and standard error using " in wh "generate" in bl " and re-issue the metan command"
exit _rc
}
cap assert "`chi2'`cornfield'`peto'`breslow'`counts'`or'`rr'`rd'`standard'`hedges'`glass'`cohen'"==""
if _rc!=0 {
di in re "Option not available without raw data counts"
exit _rc
}
if "`wgt'"!="" { local method "*" }
else {
if "`random'`randomi'"!="" {
local randomi
local random "random"
local method "D+L"
}
if "`fixed'`fixedi'"!="" {
local fixedi
local fixed "fixed"
local method "I-V"
}
cap assert ("`random'"=="") + ("`fixed'"=="")==1
if _rc!=0 {
di in re "Specify fixed or random effect/s model"
exit _rc
}
}
cap assert "`cc'"==""
if _rc!=0 {
di in re "Continuity correction not valid with unless individual counts specified"
exit _rc
}
local callalg "iv_init"
local sumstat "ES"
} /*end of 2-variable set-up */
if "`4'"!="" {
*Input is 2x2 tables: MH, Peto, IV, D+L or user defined weighting allowed
cap assert "`5'"==""
if _rc!=0 {
di in re "Wrong number of variables specified"
exit _rc
}
if "`integer'"=="" {
cap {
assert int(`1')==`1'
assert int(`2')==`2'
assert int(`3')==`3'
assert int(`4')==`4'
}
if _rc!=0 {
di in re "Non integer cell counts found"
exit _rc
}
}
cap assert ( (`1'>=0) & (`2'>=0) & (`3'>=0) & (`4'>=0) )
if _rc!=0 {
di in re "Non-positive cell counts found"
exit _rc
}
if "`cc'"!="" {
*Ensure Continuity correction is valid
if "`peto'"!="" {
di in re "Peto method not valid with continuity correction"
exit
}
*Currently, allows user to define own constant [0,1) to add to all cells
cap confirm number `cc'
if _rc!=0 {
di in re "Invalid continuity correction: specify a constant number eg metan ... , cc(0.166667)"
exit
}
cap assert (`cc'>=0) & (`cc'<1)
if _rc!=0 {
di in re "Invalid continuity correction: must be in range [0,1)"
exit
}
}
else { local cc "0.5" }
if "`peto'"=="" { local cont "cc(`cc')" }
if "`peto'"!="" { local or "or" }
capture {
assert ( ("`or'"!="")+("`rr'"!="")+("`rd'"!="") <=1 )
assert ("`fixed'"!="")+("`fixedi'"!="")+("`random'"!="")+ /*
*/ ("`randomi'"!="")+("`peto'"!="")+("`wgt'"!="") <=1
assert "`standard'`hedges'`glass'`cohen'"==""
}
if _rc!=0 {
di in re "Invalid specifications for combining trials"
exit 198
}
*Default is set at pooling RRs.
if "`or'"!="" {local sumstat "OR" }
else if "`rd'"!="" {local sumstat "RD" }
else {local sumstat "RR" }
if "`wgt'"!="" { local method "*" }
else if "`random'`randomi'"!="" {local method "D+L" }
else if "`peto'"!="" {local method "Peto"}
else if "`fixedi'"!="" {local method "I-V"}
else {local method "M-H" }
if "`peto'"!="" {local callalg "Peto"}
else {local callalg "`sumstat'"}
if ("`sumstat'"!="OR" | "`method'"=="D+L") & "`chi2'"!="" {
di in re "Chi-squared option invalid for `method' `sumstat'"
exit
}
if ("`sumstat'"!="OR" | "`method'"=="D+L" | "`method'"=="Peto" ) & "`breslow'"!="" {
di in re "Breslow-Day heterogeneity option not available for `method' `sumstat'"
exit
}
if ("`sumstat'"!="OR" & "`sumstat'"!="RR") & "`log'"!="" {
di in re "Log option not appropriate for `sumstat'"
exit
}
if "`keep'"=="" {
cap drop _SS
qui gen _SS =`1'+`2'+`3'+`4'
}
} /* end of binary variable setup */
}
if "`6'"!="" {
*Input is form N mean SD for continuous data: IV, D+L or user defined weighting allowed
cap assert "`7'"==""
if _rc!=0 {
di in re "Wrong number of variables specified"
exit _rc
}
if "`integer'"=="" {
cap assert ((int(`1')==`1') & (int(`4')==`4'))
if _rc!=0 {
di in re "Non integer sample sizes found"
exit _rc
}
}
cap assert (`1'>0 & `4'>0)
if _rc!=0 {
di in re "Non positive sample sizes found"
exit _rc
}
if "`random'`randomi'"!="" {
local randomi
local random "random"
}
if "`fixed'`fixedi'"!="" {
local fixedi
local fixed "fixed"
}
cap{
assert ("`hedges'"!="")+ ("`glass'"!="")+ ("`cohen'"!="")+ ("`standard'"!="")<=1
assert ("`random'"!="")+ ("`fixed'"!="") <=1
assert "`or'`rr'`rd'`peto'`log'`cornfield'`chi2'`breslow'`eform'"==""
}
if _rc!=0 {
di in re "Invalid specifications for combining trials"
exit 198
}
if "`standard'"!="" {
local sumstat "WMD"
local stand "none"
}
else {
if "`hedges'"!="" {local stand "hedges"}
else if "`glass'"!="" {local stand "glass" }
else {local stand "cohen"}
local sumstat "SMD"
}
local stand "standard(`stand')"
if "`wgt'"!="" { local method "*" }
else if "`random'"!="" { local method "D+L" }
else { local method "I-V" }
if "`counts'"!="" {
di in bl "Data option counts not available with continuous data"
local counts
}
if "`cc'"!="" {
di in re "Continuity correction not available with continuous data"
exit
}
local callalg "MD"
if "`keep'"=="" {
cap drop _SS
qui gen _SS =`1'+`4'
}
} /*end of 6-var set-up*/
if "`by'"!="" {
cap confirm var `by'
if _rc!=0 {
di in red "Variable `by' does not exist"
exit _rc
}
local by "by(`by')"
local nextcall "nextcall(`callalg')"
local callalg "metanby"
local sstat "sumstat(`sumstat')"
}
`callalg' `varlist' `if' `in', `by' label(`code') `keep' `table' `graph' /*
*/ method(`method') `randomi' `cont' `stand' `chi2' `cornfield' /*
*/ `log' `breslow' `eform' `wgt' `overall' `subgroup' `sgweight' /*
*/ `sortby' `saving' `xlabel' `xtick' `force' `wt' `stats' `counts' `box' /*
*/ t1(".`t1title'") t2(".`t2title'") b1(".`b1title'") b2(".`b2title'") /*
*/ `groupla' `nextcall' `sstat'
if $S_8<0 {
di in re "Insufficient data to perform this meta-analysis"
}
/*Saved results: to keep compatible with v5 have retained $S_. macros.
Now assign also to r() macros
The macro names and descriptions in the prev version are
Name v7.0 v5.0
pooled ES r(ES) $S_1*
se(ES) - not if RR or OR r(seES) $S_2
se(logES) - only if RR/OR, non-logged
r(selogES) $S_2
lower CI of pooled ES r(ci_low) $S_3
upper CI of pooled ES r(ci_upp) $S_4
Z-value for ES r(z) $S_5
p(Z) r(p_z) $S_6
chi2 heterogeneity r(het) $S_7
df (#studies-1) r(df) $S_8
p(chi2 heterogeneity) r(p_het) $S_9
Chi2 for ES (OR only) r(chi2) $S_10
p(chi2) (OR only) r(p_chi2) $S_11
Estimated tau^2(D&L only) r(tau2) $S_12
Effect measure (RR,SMD..) r(measure) -
*unless log option used..
Overall event rate [binary data; group 1)
r(tger) -
Overall event rate [binary data; group 2)
r(cger) -
*/
*return log or eform as appropriate for OR/RR
return scalar ES=$S_1
if ("`sumstat'"=="OR" | "`sumstat'"=="RR") & ("`log'"=="") { return scalar selogES=$S_2 }
else if ("`sumstat'"=="ES" & "`eform'"!="") { return scalar selogES=$S_2 }
else { return scalar seES=$S_2 }
return scalar ci_low=$S_3
return scalar ci_upp=$S_4
return scalar z=$S_5
return scalar p_z=$S_6
return scalar het=$S_7
return scalar df=$S_8
return scalar p_het=$S_9
return scalar chi2=$S_10
return scalar p_chi2=$S_11
return scalar tau2=$S_12
return local measure "`log'`sumstat'"
if ("`sumstat'"=="RR" | "`sumstat'"=="OR" | "`sumstat'"=="RD") {
return scalar tger=$S_13
return scalar cger=$S_14
}
end
program define OR
version 7.0
#delimit ;
syntax varlist(min=4 max=4 default=none numeric) [if] [in] [,
LABEL(string) SORTBY(passthru) noGRAPH noTABLE CHI2 RANDOMI CC(string)
METHOD(string) XLAbel(passthru) XTICK(passthru) FORCE CORNFIELD noKEEP
SAVING(passthru) noBOX T1(string) T2(string) B1(string) B2(string) noOVERALL
noWT noSTATS LOG BRESLOW COUNTS WGT(varlist numeric max=1) noGROUPLA ] ;
#delimit cr
qui {
tempvar a b c d use zeros r1 r2 c1 c2 t or lnor es v se ill iul ea /*
*/ va weight qhet id rawdata cont_a cont_b cont_c cont_d
tempname ernum erden R S PR PS QR QS W OR lnOR selnOR A EA VA
parse "`varlist'", parse(" ")
if "`log'"!="" { local exp }
else { local exp "exp"}
gen double `a' =`1'
gen double `b' =`2'
gen double `c' =`3'
gen double `d' =`4'
gen double `r1'=`a'+`b'
gen double `r2'=`c'+`d'
gen double `c1'=`a'+`c'
gen double `c2'=`b'+`d'
gen byte `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`r1'==.) | (`r2'==.)
replace `use'=2 if (`use'==1) & (`r1'==0 | `r2'==0 )
replace `use'=2 if (`use'==1) & (`c1'==0 | `c2'==0 )
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`counts'"!="" {
*Display raw counts
gen str20 `rawdata'= string(`a') + "/" + string(`r1') +";" + /*
*/ string(`c') + "/"+ string(`r2') if `use'!=9
replace `rawdata'= trim(`rawdata')
if "`overall'"=="" {
sum `a' if (`use'==1 | `use'==2)
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2)
local sum2=r(sum)
sum `c' if (`use'==1 | `use'==2)
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2)
local sum4=r(sum)
global MA_ODC = "`sum1'/`sum2';`sum3'/`sum4'"
}
}
else {gen str1 `rawdata'="."}
if "`method'"=="D+L" & ($S_8==0) { local method "M-H"}
*Get average event rate for each group (before any 0.5 adjustments or excluding 0-0 studies)
sum `a' if `use'<3
scalar `ernum'=r(sum)
sum `r1' if `use'<3
scalar `erden'=r(sum)
global S_13=`ernum'/`erden'
sum `c' if `use'<3
scalar `ernum'=r(sum)
sum `r2' if `use'<3
scalar `erden'=r(sum)
global S_14=`ernum'/`erden'
*Remove "uninformative" studies
replace `a'=. if `use'!=1
replace `b'=. if `use'!=1
replace `c'=. if `use'!=1
replace `d'=. if `use'!=1
replace `r1'=. if `use'!=1
replace `r2'=. if `use'!=1
gen double `t' =`r1'+`r2'
* Chi-squared test for effect
sum `a',meanonly
scalar `A'=r(sum)
gen double `ea'=(`r1'*`c1')/`t'
gen double `va'=`r1'*`r2'*`c1'*(`b'+`d')/(`t'*`t'*(`t'-1))
sum `ea',meanonly
scalar `EA'=r(sum)
sum `va',meanonly
scalar `VA'=r(sum)
global S_10=( (`A'-`EA')^2 )/`VA' /* chi2 effect value */
global S_11=chiprob(1,$S_10) /* p(chi2) */
if "`cornfield'"!="" {
*Compute Cornfield CI
gen `ill'=.
gen `iul'=.
local j=1
tempname i al aj c1j r1j r2j alold
while `j'<=_N {
if `use'[`j']==1 {
scalar `i' = 0
scalar `al' =`a'[`j']
scalar `aj' =`a'[`j']
scalar `c1j'=`c1'[`j']
scalar `r1j'=`r1'[`j']
scalar `r2j'=`r2'[`j']
scalar `alold'= .
while abs(`al'-`alold')>.001 & `al'!=. {
scalar `alold' = `al'
scalar `al'=`aj'-($ZIND)/sqrt( (1/`al') + 1/(`c1j'-`al') + /*
*/ 1/(`r1j'-`al') + 1/(`r2j'-`c1j'+`al') )
if `al'==. {
scalar `i'=`i'+1
scalar `al'=`aj'-`i'
if (`al'<0 | (`r2j'-`c1j'+`al')<0) {scalar `al'= . }
}
}
if `al'==. { scalar `al'= 0 }
replace `ill'=`log'( `al'*(`r2j'-`c1j'+`al')/((`c1j'-`al')*(`r1j'-`al')) ) in `j'
scalar `al'= `a'[`j']
scalar `alold'= .
scalar `i'= 0
while abs(`al'-`alold')>.001 & `al'!=. {
scalar `alold'= `al'
scalar `al'=`aj'+($ZIND)/sqrt( (1/`al')+ 1/(`c1j'-`al') + /*
*/ 1/(`r1j'-`al') + 1/(`r2j'-`c1j'+`al') )
if `al'==. {
scalar `i'=`i'+1
scalar `al'=`aj'+`i'
if (`al'>`r1j' | `al'>`c1j' ) { scalar `al' = . }
}
}
replace `iul'=`log'( `al'*(`r2j'-`c1j'+`al')/((`c1j'-`al')*(`r1j'-`al')) ) in `j'
}
local j=`j'+1
}
}
*Adjustment for zero cells in calcn of OR and var(OR)
gen `zeros'=1 if `use'==1 & (`a'==0 | `b'==0 | `c'==0 | `d'==0 )
gen `cont_a'=`cc'
gen `cont_b'=`cc'
gen `cont_c'=`cc'
gen `cont_d'=`cc'
replace `a'=`a'+`cont_a' if `zeros'==1
replace `b'=`b'+`cont_b' if `zeros'==1
replace `c'=`c'+`cont_c' if `zeros'==1
replace `d'=`d'+`cont_d' if `zeros'==1
replace `r1'=`r1'+(`cont_a'+`cont_b') if `zeros'==1
replace `r2'=`r2'+(`cont_c'+`cont_d') if `zeros'==1
replace `t' =`t' +(`cont_a'+`cont_b')+(`cont_c'+`cont_d') if `zeros'==1
gen double `or' =(`a'*`d')/(`b'*`c')
gen double `lnor'=log(`or')
gen double `v' =1/`a' +1/`b' +1/`c' + 1/`d'
gen double `es' =`log'(`or')
gen double `se' =sqrt(`v')
if "`cornfield'"=="" {
gen `ill' =`exp'(`lnor'-$ZIND*`se')
gen `iul' =`exp'(`lnor'+$ZIND*`se')
}
if "`method'"=="M-H" | ( "`method'"=="D+L" & "`randomi'"=="" ) {
tempname p q r s pr ps qr qs
gen double `r' =`a'*`d'/`t'
gen double `s' =`b'*`c'/`t'
sum `r', meanonly
scalar `R' =r(sum)
sum `s', meanonly
scalar `S' =r(sum)
*Calculate pooled MH- OR
scalar `OR' =`R'/`S'
scalar `lnOR'=log(`OR')
*Calculate variance/SE of lnOR and weights
gen double `p' =(`a'+`d')/`t'
gen double `q' =(`b'+`c')/`t'
gen double `pr' =`p'*`r'
gen double `ps' =`p'*`s'
gen double `qr' =`q'*`r'
gen double `qs' =`q'*`s'
sum `pr', meanonly
scalar `PR' =r(sum)
sum `ps', meanonly
scalar `PS' =r(sum)
sum `qr', meanonly
scalar `QR' =r(sum)
sum `qs', meanonly
scalar `QS' =r(sum)
scalar `selnOR'= sqrt( (`PR'/(`R'*`R') + (`PS'+`QR')/(`R'*`S') + /*
*/ `QS'/(`S'*`S'))/2 )
gen `weight'=100*`s'/`S'
*Store results in global macros, on log scale if requested
global S_1 =`log'(`OR')
global S_2 =`selnOR'
global S_3 =`exp'(`lnOR' -$ZOVE*`selnOR')
global S_4 =`exp'(`lnOR' +$ZOVE*`selnOR')
global S_5 =abs(`lnOR')/(`selnOR')
global S_6 =normprob(-abs($S_5))*2
drop `p' `q' `r' `pr' `ps' `qr' `qs'
*Calculate heterogeneity
if "`breslow'"=="" {
gen double `qhet' =( (`lnor'-`lnOR')^2 )/`v'
sum `qhet', meanonly
global S_7 =r(sum) /*Chi-squared */
global S_9 =chiprob($S_8,$S_7) /*p(chi2 het) */
}
}
if "`wgt'"!="" {
cap gen `weight'=.
udw `lnor' `v' , wgt(`wgt') `exp'
replace `weight'=`wgt'*100/$MA_W
local udwind "wgt(`wgt')"
}
else if "`method'"!="M-H" {
cap gen `weight'=.
iv `lnor' `v', method(`method') `randomi' `exp'
replace `weight'=100/( (`v'+$S_12)*($MA_W) )
}
if "`breslow'"!="" {
*Calculate heterogeneity by Breslow-Day test: need to reset zero cells and margins
if "`log'"=="" {local bexp }
else {local bexp "exp" }
replace `a'=`a'-`cont_a' if `zeros'==1
replace `b'=`b'-`cont_b' if `zeros'==1
replace `c'=`c'-`cont_c' if `zeros'==1
replace `d'=`d'-`cont_d' if `zeros'==1
replace `r1'=`r1'-(`cont_a'+`cont_b') if `zeros'==1
replace `r2'=`r2'-(`cont_c'+`cont_d') if `zeros'==1
replace `t' =`t' -(`cont_a'+`cont_b')-(`cont_c'+`cont_d') if `zeros'==1
if abs(`bexp'($S_1) - 1)<0.0001 {
gen afit = `r1'*`c1'/`t'
gen bfit = `r1'*`c2'/`t'
gen cfit = `r2'*`c1'/`t'
gen dfit = `r2'*`c2'/`t'
}
else {
tempvar sterm cterm root1 root2 afit bfit cfit dfit bresd_q
tempname qterm
scalar `qterm' = 1-`bexp'($S_1)
gen `sterm' = `r2' - `c1' + (`bexp'($S_1))*(`r1'+`c1')
gen `cterm' = -(`bexp'($S_1))*`c1'*`r1'
gen `root1' = (-`sterm' + sqrt(`sterm'*`sterm' - 4*`qterm'*`cterm'))/(2*`qterm')
gen `root2' = (-`sterm' - sqrt(`sterm'*`sterm' - 4*`qterm'*`cterm'))/(2*`qterm')
gen `afit' = `root1' if `root2'<0
replace `afit' = `root2' if `root1'<0
replace `afit' = `root1' if (`root2'>`c1') | (`root2'>`r1')
replace `afit' = `root2' if (`root1'>`c1') | (`root1'>`r1')
gen `bfit' = `r1' - `afit'
gen `cfit' = `c1' - `afit'
gen `dfit' = `r2' - `cfit'
}
gen `qhet' = ((`a'-`afit')^2)*((1/`afit')+(1/`bfit')+(1/`cfit')+(1/`dfit'))
sum `qhet', meanonly
global S_7 =r(sum) /*Het. Chi-squared */
global S_9 =chiprob($S_8,$S_7) /*p(chi2 het) */
}
replace `weight'=0 if `weight'==.
} /* End of "quietly" loop */
_disptab `es' `se' `ill' `iul' `weight' `use' `label' `rawdata', `keep' `sortby' /*
*/ `table' method(`method') sumstat(OR) `chi2' `xlabel' `xtick' `force' `graph' /*
*/ `box' `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' /*
*/ `stats' `counts' `log' `groupla' `udwind' `cornfield'
end
program define Peto
version 7.0
#delimit ;
syntax varlist(min=4 max=4 default=none numeric) [if] [in] [,
LABEL(string) ORDER(string) noGRAPH METHOD(string) CHI2 XLAbel(passthru)
XTICK(passthru) FORCE noKEEP SAVING(passthru) noBOX noTABLE SORTBY(passthru) T1(string)
T2(string) B1(string) B2(string) noOVERALL noWT noSTATS LOG COUNTS ] ;
#delimit cr
qui {
tempvar a b c d use r1 r2 t ea olesse v lnor or es se /*
*/ ill iul p weight id rawdata
tempname ernum erden OLESSE V SE P lnOR A C R1 R2
parse "`varlist'", parse(" ")
if "`log'"!="" { local exp }
else { local exp "exp"}
gen double `a' =`1' `if' `in'
gen double `b' =`2' `if' `in'
gen double `c' =`3' `if' `in'
gen double `d' =`4' `if' `in'
gen double `r1' =`a'+`b'
gen double `r2' =`c'+`d'
gen byte `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`r1'==.) | (`r2'==.)
replace `use'=2 if (`use'==1) & (`r1'==0 | `r2'==0 )
replace `use'=2 if (`use'==1) & ((`a'==0 & `c'==0 ) | (`b'==0 & `d'==0))
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`counts'"!="" {
*Display raw counts
gen str20 `rawdata'= string(`a') + "/" + string(`r1') +";" + /*
*/ string(`c') + "/"+ string(`r2') if `use'!=9
replace `rawdata'= trim(`rawdata')
if "`overall'"=="" {
sum `a' if (`use'==1 | `use'==2)
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2)
local sum2=r(sum)
sum `c' if (`use'==1 | `use'==2)
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2)
local sum4=r(sum)
global MA_ODC = "`sum1'/`sum2';`sum3'/`sum4'"
}
}
else {gen str1 `rawdata'="."}
*Get average event rate for each group (before any 0.5 adjustments or excluding 0-0 studies)
sum `a' if `use'<3
scalar `ernum'=r(sum)
sum `r1' if `use'<3
scalar `erden'=r(sum)
global S_13=`ernum'/`erden'
sum `c' if `use'<3
scalar `ernum'=r(sum)
sum `r2' if `use'<3
scalar `erden'=r(sum)
global S_14=`ernum'/`erden'
*Remove "uninformative" studies
replace `a'=. if `use'!=1
replace `b'=. if `use'!=1
replace `c'=. if `use'!=1
replace `d'=. if `use'!=1
replace `r1'=. if `use'!=1
replace `r2'=. if `use'!=1
gen double `t' =`r1'+`r2'
gen double `ea' =`r1'*(`a'+`c')/`t'
gen double `olesse'=`a'-`ea'
gen double `v' =`r1'*`r2'*(`a'+`c')*(`b'+`d')/( `t'*`t'*(`t'-1) )
gen double `lnor' =`olesse'/`v'
gen double `es' = `exp'(`lnor')
gen double `se' = 1/(sqrt(`v'))
gen double `ill' = `exp'(`lnor'-$ZIND*`se')
gen double `iul' = `exp'(`lnor'+$ZIND*`se')
gen double `p' =(`olesse')*(`olesse')/`v'
sum `olesse', meanonly
scalar `OLESSE'=r(sum)
sum `v', meanonly
scalar `V' =r(sum)
sum `p', meanonly
scalar `P' =r(sum)
scalar `lnOR' =`OLESSE'/`V'
global S_1 =`exp'(`lnOR')
global S_2 =1/sqrt(`V')
global S_3 =`exp'(`lnOR'-$ZOVE*($S_2))
global S_4 =`exp'(`lnOR'+$ZOVE*($S_2))
sum `a', meanonly
scalar `A' =r(sum)
sum `c', meanonly
scalar `C' =r(sum)
sum `r1', meanonly
scalar `R1' =r(sum)
sum `r2', meanonly
scalar `R2' =r(sum)
global S_10 =(`OLESSE'^2)/`V' /*Chi-squared effect*/
global S_11 =chiprob(1,$S_10)
global S_5 =abs(`lnOR')/($S_2)
global S_6 =normprob(-abs($S_5))*2
/*Heterogeneity */
global S_7=`P'-$S_10
global S_9 =chiprob($S_8,$S_7)
gen `weight' =100*`v'/`V'
replace `weight'=0 if `weight'==.
} /* End of quietly loop */
_disptab `es' `se' `ill' `iul' `weight' `use' `label' `rawdata' , `keep' `sortby' /*
*/ `table' method(`method') sumstat(OR) `chi2' `xlabel' `xtick' `force' `graph' `box' /*
*/ `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `counts' `log'
end
program define RR
version 7.0
#delimit ;
syntax varlist(min=4 max=4 default=none numeric) [if] [in] [,
LABEL(string) SORTBY(passthru) noGRAPH noTABLE RANDOMI METHOD(string) CC(string)
XLAbel(passthru) XTICK(passthru) FORCE noKEEP SAVING(passthru) noBOX T1(string)
T2(string) B1(string) B2(string) noOVERALL noWT noSTATS LOG COUNTS
WGT(varlist numeric max=1) ] ;
#delimit cr
qui {
tempvar a b c d use zeros r1 r2 t p r s rr lnrr es v se ill iul q /*
*/ weight id rawdata cont_a cont_b cont_c cont_d
tempname ernum erden P R S RR lnRR vlnRR
parse "`varlist'", parse(" ")
if "`log'"!="" { local exp }
else { local exp "exp"}
gen double `a' =`1'
gen double `b' =`2'
gen double `c' =`3'
gen double `d' =`4'
gen double `r1' =`a'+`b'
gen double `r2' =`c'+`d'
gen byte `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`r1'==.) | (`r2'==.)
replace `use'=2 if (`use'==1) & (`r1'==0 | `r2'==0 )
replace `use'=2 if (`use'==1) & ((`a'==0 & `c'==0 ) | (`b'==0 & `d'==0))
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`counts'"!="" {
*Display raw counts
gen str20 `rawdata'= string(`a') + "/" + string(`r1') +";" + /*
*/ string(`c') + "/"+ string(`r2') if `use'!=9
replace `rawdata'= trim(`rawdata')
if "`overall'"=="" {
sum `a' if (`use'==1 | `use'==2)
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2)
local sum2=r(sum)
sum `c' if (`use'==1 | `use'==2)
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2)
local sum4=r(sum)
global MA_ODC = "`sum1'/`sum2';`sum3'/`sum4'"
}
}
else {gen str1 `rawdata'="."}
if "`method'"=="D+L" & ($S_8==0) { local method "M-H"}
*Get average event rate for each group (before any 0.5 adjustments or excluding 0-0 studies)
sum `a' if `use'<3
scalar `ernum'=r(sum)
sum `r1' if `use'<3
scalar `erden'=r(sum)
global S_13=`ernum'/`erden'
sum `c' if `use'<3
scalar `ernum'=r(sum)
sum `r2' if `use'<3
scalar `erden'=r(sum)
global S_14=`ernum'/`erden'
*Adjustment for zero cells in calcn of OR and var(OR)
gen `zeros'=1 if `use'==1 & (`a'==0 | `b'==0 | `c'==0 | `d'==0 )
gen `cont_a'=`cc'
gen `cont_b'=`cc'
gen `cont_c'=`cc'
gen `cont_d'=`cc'
replace `a'=`a'+`cont_a' if `zeros'==1
replace `b'=`b'+`cont_b' if `zeros'==1
replace `c'=`c'+`cont_c' if `zeros'==1
replace `d'=`d'+`cont_d' if `zeros'==1
replace `r1'=`r1'+(`cont_a'+`cont_b') if `zeros'==1
replace `r2'=`r2'+(`cont_c'+`cont_d') if `zeros'==1
*Remove "uninformative" studies
replace `a'=. if `use'!=1
replace `b'=. if `use'!=1
replace `c'=. if `use'!=1
replace `d'=. if `use'!=1
replace `r1'=. if `use'!=1
replace `r2'=. if `use'!=1
gen double `t' =`r1'+`r2'
gen double `r' =`a'*`r2'/`t'
gen double `s' =`c'*`r1'/`t'
gen double `rr' =`r'/`s'
gen double `lnrr'=log(`rr')
gen double `es' =`log'(`rr')
gen double `v' =1/`a' +1/`c' - 1/`r1' - 1/`r2'
gen double `se' =sqrt(`v')
gen double `ill' =`exp'(`lnrr'-$ZIND*`se')
gen double `iul' =`exp'(`lnrr'+$ZIND*`se')
if "`method'"=="M-H" | "`method'"=="D+L" & "`randomi'"=="" {
*MH method for pooling/calculating heterogeneity in DL method
gen double `p' =`r1'*`r2'*(`a'+`c')/(`t'*`t') - `a'*`c'/`t'
sum `p', meanonly
scalar `P' =r(sum)
sum `r', meanonly
scalar `R' =r(sum)
sum `s', meanonly
scalar `S' =r(sum)
scalar `RR'=`R'/`S'
scalar `lnRR'=log(`RR')
* Heterogeneity
gen double `q' =( (`lnrr'-`lnRR')^2 )/`v'
sum `q', meanonly
global S_7 =r(sum)
global S_9 =chiprob($S_8,$S_7)
gen `weight'=100*`s'/`S'
global S_1 =`log'(`RR')
global S_2 =sqrt( `P'/(`R'*`S') )
global S_3 =`exp'(`lnRR' -$ZOVE*($S_2))
global S_4 =`exp'(`lnRR' +$ZOVE*($S_2))
global S_5 =abs(`lnRR')/($S_2)
global S_6 =normprob(-abs($S_5))*2
}
if "`wgt'"!="" {
cap gen `weight'=.
udw `lnrr' `v' , wgt(`wgt') `exp'
replace `weight'=`wgt'*100/$MA_W
local udwind "wgt(`wgt')"
}
else if "`method'"!="M-H" {
cap gen `weight'=.
iv `lnrr' `v', method(`method') `randomi' `exp'
replace `weight'=100/( (`v'+$S_12)*($MA_W) )
}
replace `weight'=0 if `weight'==.
} /* End of "quietly" loop */
_disptab `es' `se' `ill' `iul' `weight' `use' `label' `rawdata' , `keep' `sortby' /*
*/ `table' method(`method') sumstat(RR) `xlabel' `xtick' `force' `graph' `box' /*
*/ `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' /*
*/ `stats' `counts' `log' `udwind'
end
program define RD
version 7.0
#delimit ;
syntax varlist(min=4 max=4 default=none numeric) [if] [in] [,
LABEL(string) SORTBY(passthru) noGRAPH noTABLE RANDOMI METHOD(string) CC(string) noKEEP
SAVING(passthru) XLAbel(passthru) XTICK(passthru) noBOX FORCE T1(string) T2(string)
B1(string) B2(string) noOVERALL noWT noSTATS COUNTS WGT(varlist numeric max=1) ] ;
#delimit cr
qui {
tempvar a b c d use zeros r1 r2 t rd weight rdnum v se ill iul vnum q /*
*/ id w rawdata cont_a cont_b cont_c cont_d
tempname ernum erden RDNUM VNUM W
parse "`varlist'", parse(" ")
gen double `a' =`1'
gen double `b' =`2'
gen double `c' =`3'
gen double `d' =`4'
gen double `r1' =`a'+`b'
gen double `r2' =`c'+`d'
gen byte `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`r1'==.) | (`r2'==.)
replace `use'=2 if ( `use'==1) & (`r1'==0 | `r2'==0 )
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`counts'"!="" {
*Display raw counts
gen str20 `rawdata'= string(`a') + "/" + string(`r1') +";" + /*
*/ string(`c') + "/"+ string(`r2') if `use'!=9
replace `rawdata'= trim(`rawdata')
if "`overall'"=="" {
sum `a' if (`use'==1 | `use'==2)
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2)
local sum2=r(sum)
sum `c' if (`use'==1 | `use'==2)
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2)
local sum4=r(sum)
global MA_ODC = "`sum1'/`sum2';`sum3'/`sum4'"
}
}
else {gen str1 `rawdata'="."}
if "`method'"=="D+L" & ($S_8==0) { local method "M-H"}
*Get average event rate for each group (before any cont adjustments or excluding 0-0 studies)
sum `a' if `use'<3
scalar `ernum'=r(sum)
sum `r1' if `use'<3
scalar `erden'=r(sum)
global S_13=`ernum'/`erden'
sum `c' if `use'<3
scalar `ernum'=r(sum)
sum `r2' if `use'<3
scalar `erden'=r(sum)
global S_14=`ernum'/`erden'
*Remove "uninformative" studies
replace `a'=. if `use'!=1
replace `b'=. if `use'!=1
replace `c'=. if `use'!=1
replace `d'=. if `use'!=1
replace `r1'=. if `use'!=1
replace `r2'=. if `use'!=1
gen double `t' =`r1'+`r2'
gen double `rd' =`a'/`r1' - `c'/`r2'
gen `weight'=`r1'*`r2'/`t'
sum `weight',meanonly
scalar `W' =r(sum)
gen double `rdnum'=( (`a'*`r2')-(`c'*`r1') )/`t'
* Zero cell adjustments, placed here to ensure 0/n1 v 0/n2 really IS RD=0
*Adjustment for zero cells in calcn of OR and var(OR)
gen `zeros'=1 if `use'==1 & (`a'==0 | `b'==0 | `c'==0 | `d'==0 )
gen `cont_a'=`cc'
gen `cont_b'=`cc'
gen `cont_c'=`cc'
gen `cont_d'=`cc'
replace `a'=`a'+`cont_a' if `zeros'==1
replace `b'=`b'+`cont_b' if `zeros'==1
replace `c'=`c'+`cont_c' if `zeros'==1
replace `d'=`d'+`cont_d' if `zeros'==1
replace `r1'=`r1'+(`cont_a'+`cont_b') if `zeros'==1
replace `r2'=`r2'+(`cont_c'+`cont_d') if `zeros'==1
replace `t' =`t' +(`cont_a'+`cont_b')+(`cont_c'+`cont_d') if `zeros'==1
gen double `v' =`a'*`b'/(`r1'^3)+`c'*`d'/(`r2'^3)
gen double `se' =sqrt(`v')
gen double `ill' = `rd'-$ZIND*`se'
gen double `iul' = `rd'+$ZIND*`se'
if "`method'"=="M-H" | ("`method'"=="D+L" & "`randomi'"=="" ) {
sum `rdnum',meanonly
scalar `RDNUM'=r(sum)
global S_1 =`RDNUM'/`W'
gen double `q' =( (`rd'-$S_1)^2 )/`v'
sum `q', meanonly
global S_7 =r(sum)
global S_9 =chiprob($S_8,$S_7)
gen double `vnum'=( (`a'*`b'*(`r2'^3) )+(`c'*`d'*(`r1'^3))) /*
*/ /(`r1'*`r2'*`t'*`t')
sum `vnum',meanonly
scalar `VNUM'=r(sum)
global S_2 =sqrt( `VNUM'/(`W'*`W') )
replace `weight'=`weight'*100/`W'
global S_3 =$S_1 -$ZOVE*($S_2)
global S_4 =$S_1 +$ZOVE*($S_2)
global S_5 =abs($S_1)/($S_2)
global S_6 =normprob(-abs($S_5))*2
}
if "`wgt'"!="" {
udw `rd' `v' ,wgt(`wgt')
replace `weight'=`wgt'*100/$MA_W
local udwind "wgt(`wgt')"
}
else if "`method'"!="M-H" {
iv `rd' `v', method(`method') `randomi'
replace `weight'=100/( (`v'+$S_12)*($MA_W) )
}
replace `weight'=0 if `weight'==.
} /* End of "quietly" loop */
_disptab `rd' `se' `ill' `iul' `weight' `use' `label' `rawdata', `keep' `sortby' /*
*/ `table' method(`method') sumstat(RD) `xlabel' `xtick'`force' `graph' `box' /*
*/ `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `counts' `udwind'
end
program define MD
version 7.0
#delimit ;
syntax varlist(min=6 max=6 default=none numeric) [if] [in] [,
LABEL(string) SORTBY(passthru) noGRAPH METHOD(string) noKEEP SAVING(passthru) noBOX
noTABLE STANDARD(string) XLAbel(passthru) XTICK(passthru) FORCE T1(string) T2(string)
B1(string) B2(string) noOVERALL noWT noSTATS COUNTS WGT(string) ] ;
#delimit cr
qui {
tempvar n1 x1 sd1 n2 x2 sd2 use n s md v se ill iul weight id qhet rawdata
parse "`varlist'", parse(" ")
gen double `n1' =`1'
gen double `x1' =`2'
gen double `sd1'=`3'
gen double `n2' =`4'
gen double `x2' =`5'
gen double `sd2'=`6'
gen `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`n1'==.) | (`n2'==.) | (`x1'==.) | (`x2'==.) | /*
*/ (`sd1'==.) | (`sd2'==.)
replace `use'=2 if ( `use'==1) & (`n1' <2 | `n2' <2 )
replace `use'=2 if ( `use'==1) & (`sd1'<=0 | `sd2'<=0 )
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`counts'"!="" {
*Display raw counts instead of default
gen str40 `rawdata'= string(`n1') + " " + string(`x1') +" (" + string(`sd1') + /*
*/ ") ; " + string(`n2') + " " + string(`x2') +" (" + string(`sd2') +") "
replace `rawdata'= trim(`rawdata')
}
else {gen str1 `rawdata'="."}
if "`method'"=="D+L" & ($S_8==0) { local method "I-V"}
replace `n1' =. if `use'!=1
replace `x1' =. if `use'!=1
replace `sd1'=. if `use'!=1
replace `n2' =. if `use'!=1
replace `x2' =. if `use'!=1
replace `sd2'=. if `use'!=1
gen double `n' =`n1'+`n2'
if "`standard'"=="none" {
gen double `md' =`x1'-`x2'
gen double `v'=(`sd1'^2)/`n1' + (`sd2'^2)/`n2'
local prefix "W"
}
else {
gen double `s'=sqrt( ((`n1'-1)*(`sd1'^2)+(`n2'-1)*(`sd2'^2) )/(`n'-2) )
if "`standard'"=="cohen" {
gen double `md' = (`x1'-`x2')/`s'
gen double `v'= ( `n'/(`n1'*`n2') )+( (`md'^2)/(2*(`n'-2)) )
}
else if "`standard'"=="hedges" {
gen double `md' =( (`x1'-`x2')/`s' )*( 1- 3/(4*`n'-9) )
gen double `v'=( `n'/(`n1'*`n2') ) + ( (`md'^2)/(2*(`n'-3.94)) )
}
else if "`standard'"=="glass" {
gen double `md' = (`x1'-`x2')/`sd2'
gen double `v'= (`n'/(`n1'*`n2')) + ( (`md'^2)/(2*(`n2'-1)) )
}
local prefix "S"
}
gen double `se' =sqrt(`v')
gen double `ill' =`md'-$ZIND*`se'
gen double `iul' =`md'+$ZIND*`se'
if "`wgt'"!="" {
udw `md' `v' , wgt(`wgt')
gen `weight'=`wgt'*100/$MA_W
local udwind "wgt(`wgt')"
}
else {
iv `md' `v', method(`method') randomi
gen `weight'=100/( (`v'+$S_12)*($MA_W) )
}
replace `weight'=0 if `weight'==.
} /* End of quietly loop */
_disptab `md' `se' `ill' `iul' `weight' `use' `label' `rawdata', `keep' `sortby' /*
*/ `table' method(`method') sumstat(`prefix'MD) `xlabel' `xtick' `force' `graph' /*
*/ `box' `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `udwind'
end
program define iv_init
version 7.0
#delimit ;
syntax varlist(min=2 max=3 default=none numeric) [if] [in] [,
LABEL(string) SORTBY(passthru) noGRAPH METHOD(string) noKEEP SAVING(passthru) noBOX
noTABLE XLAbel(passthru) XTICK(passthru) FORCE T1(string) T2(string) B1(string)
B2(string) noOVERALL noWT noSTATS EFORM WGT(string) ] ;
#delimit cr
qui {
tempvar es se use v ill iul weight id rawdata
parse "`varlist'", parse(" ")
gen `es'=`1'
if "`eform'"!="" { local exp "exp" }
if "`3'"=="" {
gen double `se'=`2'
gen double `ill' =`exp'(`es'-$ZIND*`se' )
gen double `iul' =`exp'(`es'+$ZIND*`se' )
}
if "`3'"!="" {
gen double `se'=(`3'-`2')/($ZIND*2)
gen double `ill' =`exp'(`2')
gen double `iul' =`exp'(`3')
local var3 "var3"
}
gen double `use'=1 `if' `in'
replace `use'=9 if `use'==.
replace `use'=9 if (`es'==. | `se'==.)
replace `use'=2 if (`use'==1 & `se'<=0 )
count if `use'==1
global S_8 =r(N)-1
if $S_8<0 { exit }
if "`method'"=="D+L" & ($S_8==0) { local method "I-V"}
replace `es' =. if `use'!=1
replace `se' =. if `use'!=1
gen double `v'=(`se')^2
gen str1 `rawdata'="."
if "`wgt'"!="" {
gen `weight' = `wgt' if `use'==1
udw `es' `v', wgt(`weight') `exp'
replace `weight'=100*`wgt'/($MA_W)
local udwind "wgt(`wgt')"
}
else {
iv `es' `v', method(`method') `exp' randomi
*NB randomi necc to calculate heterogeneity
gen `weight'=100/( (`v'+$S_12)*($MA_W) )
}
replace `weight'=0 if `weight'==.
replace `es'=`exp'(`es')
} /* End of quietly loop */
_disptab `es' `se' `ill' `iul' `weight' `use' `label' `rawdata', `keep' `sortby' /*
*/`table' method(`method') sumstat(ES) `xlabel' `xtick' `force' `graph' `box' /*
*/ `saving' t1("`t1'") t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `eform' /*
*/ `var3' `udwind'
end
program define iv
version 7.0
#delimit ;
syntax varlist(min=2 max=2 default=none numeric) [if] [in] [,
METHOD(string) RANDOMI EXP ] ;
#delimit cr
tempvar stat v w qhet w2 wnew e_w e_wnew
tempname W W2 C T2 E_W E_WNEW OV vOV QHET
parse "`varlist'", parse(" ")
gen `stat'=`1'
gen `v' =`2'
gen `w' =1/`v'
sum `w',meanonly
scalar `W'=r(sum)
global S_12=0
global MA_W =`W'
if ("`randomi'"=="" & "`method'"=="D+L") { scalar `QHET'=$S_7 }
else {
gen `e_w' =`stat'*`w'
sum `e_w',meanonly
scalar `E_W'=r(sum)
scalar `OV' =`E_W'/`W'
* Heterogeneity
gen `qhet' =( (`stat'-`OV')^2 )/`v'
sum `qhet', meanonly
scalar `QHET'=r(sum)
global S_7=`QHET'
}
if "`method'"=="D+L" {
gen `w2' =`w'*`w'
sum `w2',meanonly
scalar `W2' =r(sum)
scalar `C' =`W' - `W2'/`W'
global S_12 =max(0, ((`QHET'-$S_8)/`C') )
gen `wnew' =1/(`v'+$S_12)
gen `e_wnew'=`stat'*`wnew'
sum `wnew',meanonly
global MA_W =r(sum)
sum `e_wnew',meanonly
scalar `E_WNEW'=r(sum)
scalar `OV' =`E_WNEW'/$MA_W
}
global S_1 =`exp'(`OV')
global S_2 =sqrt( 1/$MA_W )
global S_3 =`exp'(`OV' -$ZOVE*($S_2))
global S_4 =`exp'(`OV' +$ZOVE*($S_2))
global S_5 =abs(`OV')/($S_2)
global S_6 =normprob(-abs($S_5))*2
global S_9 =chiprob($S_8,$S_7)
end
program define udw
* user defined weights to combine trials
version 7.0
#delimit ;
syntax varlist(min=2 max=2 default=none numeric) [if] [in] [,
METHOD(string) EXP WGT(varlist numeric max=1) ] ;
#delimit cr
tempvar stat v w e_w varcomp qhet
tempname W E_W OV W2 Vnum V QHET
parse "`varlist'", parse(" ")
gen `stat'=`1'
gen `v' =`2'
gen `w' =`wgt' if `stat'!=.
sum `w',meanonly
scalar `W'=r(sum)
if `W'==0 {
di in re "Usable weights sum to zero: the table below will probably be nonsense"
}
global MA_W =`W'
*eff size = SIGMA(wi * thetai)/SIGMA(wi)
gen `e_w' =`stat'*`w'
sum `e_w',meanonly
scalar `E_W'=r(sum)
scalar `OV' =`E_W'/`W'
*VAR = SIGMA{wi^2 * var(thetai) }/[SIGMA(wi)]^2
sum `w',meanonly
scalar `W2'=(r(sum))^2
gen `varcomp' = `w'*`w'*`v'
sum `varcomp' ,meanonly
scalar `Vnum'=r(sum)
scalar `V' =`Vnum'/`W2'
*Heterogeneity (need to use variance weights here - BUT use ES=wgt*es/wgt, not necc var wts)
gen `qhet' =( (`stat'-`OV')^2 )/`v'
sum `qhet', meanonly
scalar `QHET'=r(sum)
global S_1 =`exp'(`OV')
global S_2 =sqrt( `V' )
global S_3 =`exp'(`OV' -$ZOVE*($S_2))
global S_4 =`exp'(`OV' +$ZOVE*($S_2))
global S_5 =abs(`OV')/($S_2)
global S_6 =normprob(-abs($S_5))*2
global S_7=`QHET'
global S_9 =chiprob($S_8,$S_7)
end
program define _disptab
version 7.0
#delimit ;
syntax varlist(min=7 max=8 default=none) [if] [in] [,
XLAbel(passthru) XTICK(passthru) FORCE noKEEP SAVING(passthru) noBOX noTABLE
noGRAPH METHOD(string) SUMSTAT(string) CHI2 T1(string) T2(string) B1(string)
B2(string) noOVERALL noWT noSTATS COUNTS LOG EFORM noGROUPLA SORTBY(string)
WGT(string) VAR3 CORNFIELD ] ;
#delimit cr
tempvar effect se lci uci weight use label tlabel rawdata id
parse "`varlist'", parse(" ")
qui {
gen `effect'=`1'
gen `se' =`2'
gen `lci' =`3'
gen `uci' =`4'
gen `weight'=`5'
gen byte `use'=`6'
format `weight' %5.1f
gen str10 `label'=""
replace `label'=`7'
global IND: displ %2.0f $IND
gen str40 `rawdata' = `8'
if "`keep'"=="" {
if ("`sumstat'"=="OR" | "`sumstat'"=="RR") & ("`log'"=="") { local ln "log"}
else { local ln }
cap drop _ES
cap drop _seES
cap drop _selogES
if "`sumstat'"!="ES" {
#delimit ;
replace _SS =. if `use'!=1; label var _SS "Sample size";
gen _ES =`effect';label var _ES "`log'`sumstat'";
gen _se`ln'ES=`se';label var _se`ln'ES "se(`ln'`log'`sumstat')";
#delimit cr
}
#delimit ;
cap drop _LCI ; cap drop _UCI; cap drop _WT;
gen _LCI =`lci'; label var _LCI "Lower CI (`log'`sumstat')";
gen _UCI =`uci'; label var _UCI "Upper CI (`log'`sumstat')";
gen _WT=`weight';label var _WT "`method' weight";
#delimit cr
}
preserve
if "`overall'"=="" {
**If overall figure requested, add an extra line to contain overall stats
qui {
local nobs1=_N+1
set obs `nobs1'
replace `effect'= ($S_1) in `nobs1'
replace `lci'=($S_3) in `nobs1'
replace `uci'=($S_4) in `nobs1'
replace `weight'=100 in `nobs1'
replace `use'=5 in `nobs1'
replace `label' = "Overall" in `nobs1'
if "`counts'"!="" { replace `rawdata'="$MA_ODC" in `nobs1' }
replace `label' = "Overall" in `nobs1'
}
}
local usetot=$S_8+1
count if `use'==2
local alltot=r(N)+`usetot'
gen `id'=_n
sort `use' `sortby' `id'
} /* End of quietly loop */
if "`table'"=="" {
qui gen str20 `tlabel'=`7' /*needs to be own label so as not to overrun!*/
if "`overall'`wt'"=="" {
local ww "% Weight"
}
if $IND!=$OVE {
global OVE: displ %2.0f $OVE
local insert "[$OVE% Conf. Interval]"
}
else { local insert "--------------------" }
di _n in gr _col(12) "Study" _col(22) "|" _col(24) "`log'" _col(28) "`sumstat'" /*
*/ _col(34) "[$IND% Conf. Interval]" _col(59) "`ww'" _n _dup(21) "-" "+" _dup(51) "-"
local i=1
while `i'<=_N {
if "`overall'`wt'"=="" { local ww=`weight'[`i'] }
else { local ww }
if (`use'[`i'])==2 {
*excluded trial
di in gr `tlabel'[`i'] _col(22) "| (Excluded)"
}
if ( (`use'[`i']==1) | (`use'[`i']==5) ) {
if (`use'[`i'])==1 {
*trial results
di in gr `tlabel'[`i'] _cont
}
else {
*overall
di in gr _dup(21) "-" "+" _dup(11) "-" "`insert'" _dup(20) "-" _n /*
*/ "`method' pooled `log'`sumstat'" _cont
}
di in gr _col(22) "|" in ye %7.3f `effect'[`i'] /*
*/ _col(35) %7.3f `lci'[`i'] " " %7.3f `uci'[`i'] _col(60) %6.2f `ww'
}
local i=`i'+1
}
di in gr _dup(21) "-" "+" _dup(51) "-"
if "`overall'"=="" {
if ("`method'"=="*" | "`var3'"!="") {
if "`method'"=="*" {
di in gr "* note: trials pooled by user defined weight `wgt'"
}
di in gr " Heterogeneity calculated by formula" _n /*
*/ " Q = SIGMA_i{ (1/variance_i)*(effect_i - effect_pooled)^2 } "
if "`var3'"!="" {
di in gr "where variance_i = ((upper limit - lower limit)/(2*z))^2 "
}
}
*Heterogeneity etc
if ( ("`sumstat'"=="OR" | "`sumstat'"=="RR") & "`log'"=="") {local h0=1}
else if ("`sumstat'"=="ES" & "`eform'"!="") {local h0=1}
else {local h0=0}
di _n in gr " Heterogeneity chi-squared = " in ye %6.2f $S_7 in gr /*
*/ " (d.f. = " in ye $S_8 in gr ") p = " in ye %4.3f $S_9
local i2=max(0, (100*($S_7-$S_8)/($S_7)) )
if $S_8<1 { local i2=. }
di in gr " I-squared (variation in `sumstat' attributable to " /*
*/ "heterogeneity) =" in ye %6.1f `i2' "%"
if "`method'"=="D+L" { di in gr " Estimate of between-study variance " /*
*/ "Tau-squared = " in ye %7.4f $S_12 }
if "`chi2'"!="" { di _n in gr " Test of OR=1: chi-squared = " in ye %4.2f /*
*/ $S_10 in gr " (d.f. = 1) p = " in ye %4.3f $S_11 }
else { di _n in gr " Test of `log'`sumstat'=`h0' : z= " in ye %6.2f $S_5 /*
*/ in gr " p = " in ye %4.3f $S_6 }
}
}
*capture only 1 trial scenario
qui {
count
if r(N)==1 {
set obs 2
replace `use'=99 in 2
replace `weight'=0 if `use'==99
}
} /*end of qui. */
if "`graph'"=="" & `usetot'>0 {
_dispgby `effect' `lci' `uci' `weight' `use' `label' `rawdata', `log' /*
*/ `xlabel' `xtick' `force' sumstat(`sumstat') `saving' `box' t1("`t1'") /*
*/ t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `counts' `eform' /*
*/ `groupla' `cornfield'
}
restore
end
program define metanby
version 7.0
#delimit ;
syntax varlist(min=2 max=6 default=none numeric) [if] [in] [, BY(string)
LABEL(string) SORTBY(string) noGRAPH noTABLE noKEEP NEXTCALL(string)
METHOD(string) SUMSTAT(string) RANDOMI WGT(passthru) noSUBGROUP SGWEIGHT
CORNFIELD CHI2 CC(passthru) STANDARD(passthru) noOVERALL LOG EFORM BRESLOW
XLAbel(passthru) XTICK(passthru) FORCE SAVING(passthru) T1(string) T2(string)
B1(string) B2(string) noWT noSTATS COUNTS noBOX noGROUPLA ] ;
#delimit cr
if ("`subgroup'"!="" & "`overall'`sgweight'"!="") { local wt "nowt" }
tempvar use by2 newby r1 r2 rawdata effect se lci uci weight wtdisp /*
*/ hetc hetdf hetp i2 tau2 tsig psig expand tlabel id
qui {
gen `use'=1 `if' `in'
replace `use'=9 if `use'==.
gen str1 `rawdata'="."
tokenize `varlist'
if ("`nextcall'"=="RR" | "`nextcall'"=="OR" | "`nextcall'"=="RD" |"`nextcall'"=="Peto" ) {
*Sort out r1 & r2 for 2x2 table: might be needed in counts and mh/use
gen `r1' = `1'+`2'
gen `r2' = `3'+`4'
replace `use'=2 if ((`use'==1) & (`r1'==0 | `r2'==0 ))
replace `use'=2 if ((`use'==1) & ((`1'+`3'==0) | (`2'+`4'==0) ) & "`nextcall'"!="RD")
replace `use'=9 if (`r1'==.) | (`r2'==.)
if "`counts'"!="" {
*create new variable with count data (if requested)
replace `rawdata'= trim( string(`1') + "/" + string(`r1') +";" + /*
*/ string(`3') + "/"+ string(`r2') ) if `use'!=9
}
}
if "`nextcall'"=="MD" {
*Sort out n1 & n2
replace `use'=9 if (`1'==.) | (`2'==.) | (`3'==.) | (`4'==.) | (`5'==.) | (`6'==.)
replace `use'=2 if ( `use'==1) & (`1' <2 | `4' <2 )
replace `use'=2 if ( `use'==1) & (`3'<=0 | `6'<=0 )
}
if "`nextcall'"=="iv_init" {
replace `use'=9 if (`1'==. | `2'==.)
if "`3'"=="" {
replace `use'=2 if (`use'==1 & `2'<=0 )
}
else {
replace `use'=9 if (`3'==.)
replace `use'=2 if ( `2'>`1' | `3'<`1' | `3'<`2')
}
}
if (("`sumstat'"=="OR" | "`sumstat'"=="RR") & "`log'"=="" ) {local h0=1}
else if ("`sumstat'"=="ES" & "`eform'"!="") {local h0=1}
else {local h0=0}
if "`eform'"!="" { local exp "exp" }
*Get the individual trial stats
`nextcall' `varlist' if `use'==1, nograph notable method(`method') `randomi' /*
*/ label(`label') `wgt' `cornfield' `chi2' `cc' `standard' `log' `eform' `breslow'
if $S_8<0 {
*no trials - bomb out
exit
}
local nostud=$S_8
*need to calculate from variable itself if only 2 variables (ES, SE(ES) syntax used)
if "`sumstat'"=="ES" {
gen `effect'=`exp'(`1')
if "`3'"=="" {
gen `se'=`2'
}
else {
gen `se'=.
local var3 "var3"
}
}
else {
gen `effect'=_ES
if `h0'<0.01 { gen `se'=_seES }
else { gen `se'=_selogES }
}
gen `lci'=_LCI
gen `uci'=_UCI
gen `weight'=_WT
*put overall weight into var if requested
if ("`sgweight'"=="" & "`overall'"=="" ) {
gen `wtdisp'=_WT
}
else {
gen `wtdisp'=.
}
gen `id'=_n
sort `by' `sortby' `id'
*Keep only neccesary data (have to put preserve here in order to keep _ES etc)
preserve
drop if `use'==9
*Can now forget about the if/in conditions specified: unnecc rows have been removed
*Keep tger and cger here (otherwise it ends up in last subgroup only)
if ("`sumstat'"=="OR" | "`sumstat'"=="RR" | "`sumstat'"=="RD" ) {
local tger=$S_13
local cger=$S_14
}
*subgroup component of heterogeneity
gen `hetc'=.
gen `hetdf'=.
gen `hetp'=.
gen `i2'=.
gen `tau2'=.
gen `tsig'=.
gen `psig'=.
*Convert "by" variable to numeric if its a string variable
cap confirm numeric var `by'
if _rc>0 {
encode `by', gen(`by2')
drop `by'
rename `by2' `by'
}
*Create new "by" variable to take on codes 1,2,3..
gen `newby'=(`by'>`by'[_n-1])
replace `newby'=1+sum(`newby')
local ngroups=`newby'[_N]
if "`overall'"=="" {
*If requested, add an extra line to contain overall stats
local nobs1=_N+1
set obs `nobs1'
replace `use'=5 in `nobs1'
replace `newby'=`ngroups'+1 in `nobs1'
replace `effect'= ($S_1) in `nobs1'
replace `lci'=($S_3) in `nobs1'
replace `uci'=($S_4) in `nobs1'
*Put cell counts in subtotal row
if ("`counts'"!="" & "`nextcall'"!="MD") {
*put up overall binary count data
sum `1' if (`use'==1 | `use'==2)
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2)
local sum2=r(sum)
sum `3' if (`use'==1 | `use'==2)
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2)
local sum4=r(sum)
replace `rawdata'= "`sum1'/`sum2';`sum3'/`sum4'" in `nobs1'
}
replace `hetc' =($S_7) in `nobs1'
replace `hetdf'=($S_8) in `nobs1'
replace `hetp' =($S_9) in `nobs1'
replace `i2'=max(0, ( 100*($S_7-$S_8))/($S_7) ) in `nobs1'
if $S_8<1 { replace `i2'=. in `nobs1' }
replace `tau2' =$S_12 in `nobs1'
replace `se'=$S_2 in `nobs1'
if "`chi2'"!="" {
replace `tsig'=$S_10 in `nobs1'
replace `psig'=$S_11 in `nobs1'
local z=$S_5
local pz=$S_6
}
else {
replace `tsig'=$S_5 in `nobs1'
replace `psig'=$S_6 in `nobs1'
local echi2 =$S_10
local pchi2=$S_11
}
replace `label' = "Overall" in `nobs1'
if "`sgweight'"=="" { replace `wtdisp'=100 in `nobs1' }
}
*Create extra 2 or 3 lines per bygroup: one to label, one for gap
*and one for overall effect size (unless no subgroup combining is done)
sort `newby' `use' `sortby' `id'
by `newby': gen `expand'=1 + 2*(_n==1) + (_n==1 & "`subgroup'"=="")
replace `expand'=1 if `use'==5
expand `expand'
gsort `newby' -`expand' `use' `sortby' `id'
by `newby': replace `use'=0 if `expand'>1 & _n==2 /* row for by label */
by `newby': replace `use'=4 if `expand'>1 & _n==3 /* row for blank line */
by `newby': replace `use'=3 if `expand'>1 & _n==4 /* (if specified) row to hold subgp effect sizes */
* blank out effect sizes in new rows
replace `effect'=. if `expand'>1 & `use'!=1
replace `lci'=. if `expand'>1 & `use'!=1
replace `uci'=. if `expand'>1 & `use'!=1
replace `weight' =. if `expand'>1 & `use'!=1
replace `rawdata' ="." if `expand'>1 & `use'!=1
*Perform subgroup analyses
local j=1
while `j'<=`ngroups' {
if "`subgroup'"=="" {
*First ensure the by() category has any data
count if (`newby'==`j' & `use'==1)
if r(N)==0 {
*No data in subgroup=> fill variables with missing and move on
replace `effect'=. if (`use'==3 & `newby'==`j')
replace `lci'=. if (`use'==3 & `newby'==`j')
replace `uci'=. if (`use'==3 & `newby'==`j')
replace `wtdisp'=0 if `newby'==`j'
replace `weight'=0 if `newby'==`j'
replace `hetc'=. if `newby'==`j'
replace `hetdf'=. if `newby'==`j'
replace `hetp'=. if `newby'==`j'
replace `i2'=. if `newby'==`j'
replace `tsig'=. if `newby'==`j'
replace `psig'=. if `newby'==`j'
replace `tau2'=. if `newby'==`j'
}
else {
`nextcall' `varlist' if (`newby'==`j' & `use'==1) , nograph /*
*/ notable label(`label') method(`method') `randomi' `wgt' `cornfield' `chi2' /*
*/ `cc' `standard' `log' `eform' `breslow'
replace `effect'=($S_1) if `use'==3 & `newby'==`j'
replace `lci'=($S_3) if `use'==3 & `newby'==`j'
replace `uci'=($S_4) if `use'==3 & `newby'==`j'
*Put within-subg weights in if nooverall or sgweight options specified
if ("`overall'`sgweight'"!="" ) {
replace `wtdisp'=_WT if `newby'==`j'
replace `wtdisp'=100 if (`use'==3 & `newby'==`j')
}
else {
qui sum `wtdisp' if (`use'==1 & `newby'==`j')
replace `wtdisp'=r(sum) if (`use'==3 & `newby'==`j')
}
sum `weight' if `newby'==`j'
replace `weight'= r(sum) if `use'==3 & `newby'==`j'
replace `hetc' =($S_7) if `use'==3 & `newby'==`j'
replace `hetdf'=($S_8) if `use'==3 & `newby'==`j'
replace `hetp' =($S_9) if `use'==3 & `newby'==`j'
replace `i2'=max(0, ( 100*($S_7-$S_8))/($S_7) ) if `use'==3 & `newby'==`j'
if $S_8<1 { replace `i2'=. if `use'==3 & `newby'==`j' }
if "`chi2'"!="" {
replace `tsig'=($S_10) if `use'==3 & `newby'==`j'
replace `psig'=($S_11) if `use'==3 & `newby'==`j'
}
else {
replace `tsig'=($S_5) if `use'==3 & `newby'==`j'
replace `psig'=($S_6) if `use'==3 & `newby'==`j'
}
if "`method'"=="D+L" {
replace `tau2' =($S_12) if `use'==3 & `newby'==`j'
}
}
*Whether data or not - put cell counts in subtotal row if requested (will be 0/n1;0/n2 or blank if all use>1)
if "`counts'"!="" {
*don't put up anything for MDs:
*1 Cochrane just put up N_gi. Not sure whether weighted mean should be in..
*2 justifying N_gi is tedious!
if "`nextcall'"!="MD" {
sum `1' if (`use'==1 | `use'==2) & (`newby'==`j')
local sum1=r(sum)
sum `r1' if (`use'==1 | `use'==2) & (`newby'==`j')
local sum2=r(sum)
sum `3' if (`use'==1 | `use'==2) & (`newby'==`j')
local sum3=r(sum)
sum `r2' if (`use'==1 | `use'==2) & (`newby'==`j')
local sum4=r(sum)
replace `rawdata'= "`sum1'/`sum2';`sum3'/`sum4'" if (`use'==3 & `newby'==`j')
}
}
}
*Label attatched (if any) to byvar
local lbl: value label `by'
sum `by' if `newby'==`j'
local byvlu=r(mean)
if "`lbl'"=="" { local lab "`by'==`byvlu'" }
else { local lab: label `lbl' `byvlu' }
replace `label' = "`lab'" if ( `use'==0 & `newby'==`j')
replace `label' = "Subtotal" if ( `use'==3 & `newby'==`j')
local j=`j'+1
}
} /*End of quietly loop*/
*Put table up (if requested)
sort `newby' `use' `sortby' `id'
if "`table'"=="" {
qui gen str20 `tlabel'=`label'
if "`overall'`wt'"=="" {
local ww "% Weight"
}
di _n in gr _col(12) "Study" _col(22) "|" _col(24) "`log'" _col(28) "`sumstat'" /*
*/ _col(34) "[$IND% Conf. Interval]" _col(59) "`ww'"
di _dup(21) "-" "+" _dup(51) "-"
*legend for pooled confidence intervals
local i=1
while `i'<= _N {
if (`use'[`i'])==0 {
*by label
di _col(6) in gr `tlabel'[`i']
}
if "`overall'`wt'"=="" { local ww=`wtdisp'[`i'] }
else { local ww }
if (`use'[`i'])==1 {
*trial results
di in gr `tlabel'[`i'] _col(22) "| " in ye %7.3f `effect'[`i'] /*
*/ _col(35) %7.3f `lci'[`i'] " " %7.3f `uci'[`i'] _col(60) %6.2f `ww'
}
if (`use'[`i'])==2 {
*excluded trial
di in gr `tlabel'[`i'] _col(22) "| (Excluded)"
}
if ((`use'[`i']==3) & "`subgroup'"=="") | (`use'[`i']==5) {
*Subgroup effect size or overall effect size
if (`use'[`i'])==3 {
di in gr " Sub-total" _col(22) "|"
}
if (`use'[`i'])==5 {
if $IND!=$OVE {
local insert "[$OVE% Conf. Interval]"
}
di in gr "Overall" _col(22) "|" _col(34) "`insert'"
}
if "`ww'"=="." { local ww }
di in gr " `method' pooled `log'`sumstat'" _col(22) "| " in ye %7.3f /*
*/ `effect'[`i'] _col(35) %7.3f `lci'[`i'] " " %7.3f `uci'[`i'] _col(60) %6.2f `ww'
if (`use'[`i'])==5 {
di in gr _dup(21) "-" "+" _dup(51) "-"
}
}
if (`use'[`i'])==4 {
*blank line separator (need to put line here in case nosubgroup was selected)
di in gr _dup(21) "-" "+" _dup(51) "-"
}
local i=`i'+1
}
*Skip next bits if nooverall AND nosubgroup
if ("`subgroup'"=="" | "`overall'"=="") {
*part 2: user defined weight notes and heterogeneity
if ("`method'"=="*" | "`var3'"!="") {
if "`method'"=="*" {
di in gr "* note: trials pooled by user defined weight `wgt'"
}
di in bl " Heterogeneity calculated by formula" _n /*
*/ " Q = SIGMA_i{ (1/variance_i)*(effect_i - effect_pooled)^2 } "
if "`var3'"!="" {
di in bl "where variance_i = ((upper limit - lower limit)/(2*z))^2 "
}
}
di in gr _n "Test(s) of heterogeneity:" _n _col(16) "Heterogeneity degrees of"
di in gr _col(18) "statistic freedom P I-squared**" _cont
if "`method'"=="D+L" { di in gr " Tau-squared" }
di
local i=1
while `i'<= _N {
if ("`subgroup'"=="" & (`use'[`i'])==0) | ( (`use'[`i'])==5) {
di in gr _n `tlabel'[`i'] _cont
}
if ( ((`use'[`i'])==3) | ((`use'[`i'])==5) ) {
di in ye _col(20) %6.2f `hetc'[`i'] _col(35) %2.0f `hetdf'[`i'] /*
*/ _col(43) %4.3f `hetp'[`i'] _col(51) %6.1f `i2'[`i'] "%" _cont
if "`method'"=="D+L" {
di in ye " " %7.4f `tau2'[`i'] _cont
}
if (`use'[`i']==5) & ("`subgroup'"=="") {
qui sum `hetc' if `use'==3
local btwghet = (`hetc'[`i']) -r(sum)
local df = `ngroups'-1
di _n in gr "Overall Test for heterogeneity between sub-groups : " _n /*
*/ in ye _col(20) %6.2f `btwghet' _col(35) %2.0f `df' _col(43) %4.3f (chiprob(`df',`btwghet'))
}
}
local i=`i'+1
}
di _n in gr "** I-squared: the variation in `sumstat' attributable to heterogeneity)" _n
*part 3: test statistics
di _n in gr "Significance test(s) of `log'`sumstat'=`h0'"
local i=1
while `i'<= _N {
if ("`subgroup'"=="" & (`use'[`i'])==0) | ( (`use'[`i'])==5) {
di _n in gr `tlabel'[`i'] _cont
}
if ( ((`use'[`i'])==3) | ((`use'[`i'])==5) ) {
if "`chi2'"!="" {
di in gr _col(20) "chi-squared = " in ye %5.2f `tsig'[`i'] /*
*/ in gr _col(35) " (d.f. = 1) p = " in ye %4.3f `psig'[`i'] _cont
}
else {
di in gr _col(23) "z= " in ye %5.2f `tsig'[`i'] _col(35) in gr /*
*/ " p = " in ye %4.3f `psig'[`i'] _cont
}
}
local i=`i'+1
}
di _n in gr _dup(73) "-"
}
} /* end of table display */
if "`overall'"=="" {
*need to return overall effect to $S_1 macros and so on...
global S_1=`effect'[_N]
global S_2=`se'[_N]
global S_3=`lci'[_N]
global S_4=`uci'[_N]
global S_7=`hetc'[_N]
global S_8=`hetdf'[_N]
global S_9=`hetp'[_N]
if "`chi2'"!="" {
global S_10=`tsig'[_N]
global S_11=`psig'[_N]
global S_5=`z'
global S_6=`pz'
}
else {
global S_5=`tsig'[_N]
global S_6=`psig'[_N]
global S_10=`echi2'
global S_11=`pchi2'
}
global S_12=`tau2'[_N]
if ("`sumstat'"=="OR" | "`sumstat'"=="RR" | "`sumstat'"=="RD" ) {
global S_13=`tger'
global S_14=`cger'
}
}
else {
#delimit ;
global S_1 "."; global S_2 "."; global S_3 "."; global S_4 ".";
global S_5 "."; global S_6 "."; global S_7 "."; global S_8= `nostud';
global S_9 "."; global S_10 ".";global S_11 ".";global S_12 ".";
global S_13 ".";global S_14 ".";
#delimit cr
}
if "`graph'"=="" {
_dispgby `effect' `lci' `uci' `weight' `use' `label' `rawdata' `wtdisp', /*
*/ `log' `xlabel' `xtick' `force' sumstat(`sumstat') `saving' `box' t1("`t1'") /*
*/ t2("`t2'") b1("`b1'") b2("`b2'") `overall' `wt' `stats' `counts' `eform' /*
*/ `groupla' `cornfield'
}
restore
if "`keep'"=="" {
qui{
if ("`sumstat'"=="OR" | "`sumstat'"=="RR") & ("`log'"=="") { local ln "log"}
else { local ln }
cap drop _ES
cap drop _seES
cap drop _selogES
if "`sumstat'"!="ES" {
#delimit ;
replace _SS =. if `use'!=1; label var _SS "Sample size";
gen _ES =`effect';label var _ES "`log'`sumstat'";
gen _se`ln'ES=`se';label var _se`ln'ES "se(`ln'`log'`sumstat')";
#delimit cr
}
#delimit ;
cap drop _LCI ; cap drop _UCI; cap drop _WT;
gen _LCI =`lci'; label var _LCI "Lower CI (`log'`sumstat')";
gen _UCI =`uci'; label var _UCI "Upper CI (`log'`sumstat')";
#delimit cr
*correct weight if subgroup weights given
if ("`sgweight'"=="" & "`overall'"=="" ) { gen _WT=`weight' }
else if "`subgroup'"=="" & ("`overall'`sgweight'"!="" ) {
tempvar tempsum
by `by': gen `tempsum'=sum(`weight')
by `by': replace `tempsum'=`tempsum'[_N]
gen _WT=`weight'*100/`tempsum'
local sg "(subgroup) "
}
cap label var _WT "`method' `sg'% weight"
}
}
end
program define _dispgby
version 7.0
#delimit ;
syntax varlist(min=6 max=8 default=none ) [if] [in] [,
LOG XLAbel(string) XTICK(string) FORCE SAVING(string) noBOX SUMSTAT(string)
T1(string) T2(string) B1(string) B2(string) noOVERALL noWT noSTATS COUNTS EFORM
noGROUPLA CORNFIELD ];
#delimit cr
tempvar effect lci uci weight wtdisp use label tlabel id yrange xrange Ghsqrwt rawdata
parse "`varlist'", parse(" ")
qui {
gen `effect'=`1'
gen `lci' =`2'
gen `uci' =`3'
gen `weight'=`4'
gen byte `use'=`5'
*Use is now coded:
*0: blank line, except for text containing "by"
*1: trial
*2: excluded trial
*3. subgroup effect (in which case `label' or `6' is to contain the name of the subgroup)
*4. blank line
*5. overall effect
*9. missed/not considered
*As before effect sizes are held elsewhere
gen str10 `label'=""
replace `label'=`6'
count if (`use'==1 | `use'==2)
local ntrials=r(N)
count if (`use'>=0 & `use'<=5)
local ymax=r(N)
gen `id'=`ymax'-_n+1 if `use'<9
gen str40 `rawdata' = `7'
compress `rawdata'
if "`8'"!="" {
gen `wtdisp'=`8'
}
else {
gen `wtdisp'=`weight'
}
format `wtdisp' %5.1f
sum `lci'
local Gxlo=r(min) /* minimum of CIs*/
sum `uci'
local Gxhi=r(max) /* maximum of CIs*/
local h0=0
if (("`sumstat'"=="OR" | "`sumstat'"=="RR") & ("`log'"=="")) | ("`eform'"!="") {
local h0=1
local Glog "xlog"
local xlog "log"
local xexp "exp"
replace `lci'=1e-9 if `lci'<1e-8
replace `lci'=1e9 if `lci'>1e8 & `lci'!=.
replace `uci'=1e-9 if `uci'<1e-8
replace `uci'=1e9 if `uci'>1e8 & `uci'!=.
if `Gxlo'<1e-8 {local Gxlo=1e-8}
if `Gxhi'>1e8 {local Gxhi=1e8}
}
if "`cornfield'"!="" {
replace `lci'=`log'(1e-9) if ( (`lci'==. | `lci'==0) & (`effect'!=. & `use'==1) )
replace `uci'=`log'(1e9) if ( (`uci'==.) & (`effect'!=. & `use'==1) )
}
local flag1=0
if "`xtick'"!="" {
* capture inappropriate tick
cap assert ("`xlog'"=="" ) | /*
*/ ( ( min(`xtick' ,`Gxhi')>1e-8 ) & (max(`xtick' ,`Gxlo')<1e8) )
if _rc!=0 {
local flag1=10
local xtick "`h0'"
}
}
else {
local xtick "`h0'"
}
if "`xlabel'"!="" {
* capture inappropriate label
cap {
assert ("`xlog'"=="" ) | /*
*/ ( ( min(`xlabel',`Gxhi')>1e-8 ) & (max(`xlabel',`Gxlo')<1e8) )
}
if _rc!=0 {
local flag1=10
local xlabel
}
else {
if "`force'"!="" {
parse "`xlabel'", parse(",")
if "`3'"!="" {
local Gxlo=`h0'
local Gxhi=`h0'
}
}
}
}
if "`xlabel'"=="" | (`flag1'>1) {
local Gmodxhi=max( abs(`xlog'(`Gxlo')),abs(`xlog'(`Gxhi')))
if `Gmodxhi'==. {local Gmodxhi=2}
local Gxlo=`xexp'(-`Gmodxhi')
local Gxhi=`xexp'( `Gmodxhi')
local xlabel "`Gxlo',`h0',`Gxhi'"
}
local Gxlo=`xlog'(min(`xlabel',`xtick',`Gxlo'))
local Gxhi=`xlog'(max(`xlabel',`xtick',`Gxhi'))
local Gxlo1=`Gxlo'-0.1*`Gxhi'
local Gxrange=(`Gxhi'-`Gxlo')
local Gyhi=`id'[1]
local Gxh20=`Gxhi'+`Gxrange'*0.2
local Gxh40=`Gxhi'+`Gxrange'*0.4
local Gxh60=`Gxhi'+`Gxrange'*0.6
local Gxh80=`Gxhi'+`Gxrange'*0.8
local Gxh100=`Gxhi'+`Gxrange'*1.0
gen `xrange'=`xexp'(`Gxlo') in 1
*If user wants no counts (c), stats (s) or weights (w) , use entire window for trial plots
if ("`stats'"!="" & "`wt'"!="" & "`counts'"=="" ) {
local Txhi=`Gxhi'
}
*If user wants s&w (default) or c&w use Gxh60 (60% of graph range) for figures
if (("`stats'"=="" & "`wt'"=="" & "`counts'"=="" ) | ("`stats'"!="" & "`wt'"=="" & "`counts'"!="" )) {
local Txhi=`Gxh60'
}
*If user wants s or c alone use Gxh40 (40% of range)
if (("`stats'"=="" & "`wt'"!="" & "`counts'"=="" ) | ("`stats'"!="" & "`wt'"!="" & "`counts'"!="" )) {
local Txhi=`Gxh40'
}
*If user wants w alone use Gxh20
if ("`stats'"!="" & "`wt'"=="" & "`counts'"=="" ) {
local Txhi=`Gxh20'
}
*If user wants s&c use Gxh80
if ("`stats'"=="" & "`wt'"!="" & "`counts'"!="" ) {
local Txhi=`Gxh80'
}
*If user wants all 3 use Gxh100
if ("`stats'"=="" & "`wt'"=="" & "`counts'"!="" ) {
local Txhi=`Gxh100'
}
replace `xrange'=`xexp'(`Txhi') in 2
gen `yrange'=0 in 1
replace `yrange'=`Gyhi'+2 in 2
cap label drop tmpyl
cap label drop tmpxl
* Study legend now removed
label define tmpyl 0 " "
label define tmpxl `h0' " "
label values `yrange' tmpyl
label values `xrange' tmpxl
*Label x-axis and top right *if stats requested
if "`sumstat'"=="OR" {local sscale "`log' Odds ratio"}
else if "`sumstat'"=="RR" {local sscale "`log' Risk ratio"}
else if "`sumstat'"=="RD" {local sscale "Risk difference"}
else if "`sumstat'"=="WMD" {local sscale "Mean difference"}
else if "`sumstat'"=="SMD" {local sscale "Standardised mean difference"}
else if ("`sumstat'"=="ES" & "$MA_ESLA"!="") { local sscale "$MA_ESLA" }
else if "`sumstat'"=="ES" {local sscale "Effect size"}
if "`t1'"=="." {local t1 }
else {
local t1=substr("`t1'",2,.)
local t1 "t1(`t1')"
}
if "`t2'"=="." {local t2 }
else {
local t2=substr("`t2'",2,.)
local t2 "t2(`t2')"
}
if "`b2'"=="." { local b2 "`sscale'"}
else {
*Revise position of b2 title: graph command doesn't put it in the right place
local b2=substr("`b2'",2,.)
}
if "`saving'"!="" {local saving "saving(`saving')" }
else { local saving }
gph open, `saving'
if "`b1'"=="." {
graph `yrange' `xrange', s(i) xli(`h0') `Glog' xlabel(`h0') /*
*/ noaxis yla(0) gap(10) `t1' `t2' b2(" ")
}
else {
graph `yrange' `xrange', s(i) xli(`h0') `Glog' xlabel(`h0') /*
*/ noaxis yla(0) gap(10) `t1' `t2' b1(" ") b2(" ")
}
local r5=r(ay)
local r6=r(by)
local r7=r(ax)
local r8=r(bx)
local Aytexs=($MA_FTSI)*max(200, min(700, (600-20*(`ymax'-15)) ) )
local Axtexs=($MA_FTSI)*max(130, min(360,(0.6*`Aytexs')) )
gph font `Aytexs' `Axtexs'
gph pen 1
local Axh0 =`r7'*(`xlog'(`h0'))+`r8'
local Axlo =`r7'*(`Gxlo') +`r8'
local Axloe =`r7'*(`Gxlo') +`r8' -1500
local Axhi =`r7'*(`Gxhi') +`r8'
local Axhie =`r7'*(`Gxhi') +`r8' + 1500*( ("`wt'"=="") | ("`stats'"=="") | ("`counts'"!="") )
local Axh20 =`r7'*(`Gxh20') +`r8'
local Axh40 =`r7'*(`Gxh40') +`r8'
local Axh60 =`r7'*(`Gxh60') +`r8'
local Axh80 =`r7'*(`Gxh80') +`r8'
local Axh100=`r7'*(`Gxh100')+`r8'
*add x-axis line and label manually
gph line `r6' `Axloe' `r6' `Axhie'
local yb=`r6'+1000
gph line `r6' `Axh0' `yb' `Axh0'
local yb=`r6'+2400+max(0,(1000-100*`ymax'))
gph text `yb' `Axh0' 0 0 `b2'
if "`b1'"!="." {
local yb=`r6'+3000+max(0,(1200-100*`ymax'))
local b1=substr("`b1'",2,.)
if substr("`b1'",1,3)=="*I:" {
*"Favours ..." labels
local flab= substr("`b1'",4,.)
tokenize "`flab'" , parse(*)
if "`1'"!="" {
local xt=`Axh0'-1000
gph text `yb' `xt' 0 1 `1'
}
if "`3'"!="" {
local xt=`Axh0'+1000
gph text `yb' `xt' 0 -1 `3'
}
}
else {
gph text `yb' `Axh0' 0 0 `b1'
}
}
*add xtick & xlabel manually
tokenize "`xlabel'", parse(,)
while "`1'"!="" {
local x=`1'
if (`x'>10e8) { local x="10e8" }
if ((`x'<=-10e8) & (`h0'==0)) { local x="-10e8" }
local Ax=`r7'*(`xlog'(`x')) +`r8'
local Ayh=`r6'
local Ayl=`r6'+400
local Ayt=`r6'+1700
gph line `Ayh' `Ax' `Ayl' `Ax'
if ((`x'<10e-6) & (`h0'==1)) { local x="0.00000" }
if ((`x'<10e4) & (`x'>-10e4)) { local x=substr("`x'",1,7) }
gph text `Ayt' `Ax' 0 0 `x'
mac shift 2
}
tokenize "`xtick'", parse(,)
while "`1'"!="" {
local Ax=`r7'*(`xlog'(`1')) + `r8'
local Ayh=`r6'
local Ayl=`r6'+400
local Ayt=`r6'+1000
gph line `Ayh' `Ax' `Ayl' `Ax'
mac shift 2
}
*Add legend
local Ayhi =(`Gyhi'+2)*`r5'+`r6'
local Axtexh=50+max(0,min(2400,`ymax'*60-700))
local Ayhead=`r5'*(`ymax'+1) + `r6'
gph text `Ayhead' `Axtexh' 0 -1 $S_TX
/*Order of text: stats, counts, weights (where selected)*/
*New positionings reduce gaps between stats/weight/counts
if ("`stats'"=="") {
*Align stats first at Axhi. Use prev line if `sscale' too long (>12chrs)
* local lenssc=length("`sscale'")
* if `lenssc'<13 {
* gph text `Ayhead' `Axhi' 0 -1 `sscale' ($IND% CI)
* }
* else {
local Ayhead2=`r5'*(`ymax'+1.65) + `r6'
gph text `Ayhead2' `Axhi' 0 -1 `sscale'
gph text `Ayhead' `Axhi' 0 -1 ($IND% CI)
* }
}
if ("`counts'"!="") {
*Align counts 2nd if both stats & weight
if ("`stats'"=="") {
local Ax2hi1=`Axh20'+(`Axh60'-`Axh20')*(max(0.4, min(1,2-0.07*`ymax') ) )
local Ax2hi2=`Axh20'+(`Axh80'-`Axh20')*(max(0.4, min(1,2-0.07*`ymax') ) )
*local Ax2hi1=`Axh60'
*local Ax2hi2=`Axh80'
}
else {
local Ax2hi1=`Axhi'+(`Axh20'-`Axhi')*(max(0.4, min(1,2-0.07*`ymax') ) )
local Ax2hi2=`Axhi'+(`Axh40'-`Axhi')*(max(0.4, min(1,2-0.07*`ymax') ) )
*local Ax2hi1=`Axh20'
*local Ax2hi2=`Axh40'
}
local yt=`Ayhead'-max(200, min(1000,1600-40*`ymax'))
gph text `yt' `Ax2hi1' 0 0 No. of events
* gph font 500 270
gph text `Ayhead' `Ax2hi1' 0 1 $MA_G1L
gph text `Ayhead' `Ax2hi2' 0 1 $MA_G2L
}
if ("`wt'"=="") {
if ("`stats'"=="" & "`counts'"!="") {
*local Ax3hi=`Axh100'
local Ax3hi=`Axh20'+(`Axh100'-`Axh20')*(max(0.4, min(1,2-0.07*`ymax') ) )
}
else if ("`stats'"!="" & "`counts'"=="") {
local Ax3hi=`Axh20'
}
else {
*local Ax3hi=`Axh60'
local Ax3hi=`Axh20'+(`Axh60'-`Axh20')*(max(0.4, min(1,2-0.07*`ymax') ) )
}
gph text `Ayhead' `Ax3hi' 0 1 % Weight
}
gen `Ghsqrwt'=0.5*sqrt(`weight')/2
local flag=0
while `flag'<1 {
cap assert `Ghsqrwt'<0.5 if (`use'==1)
if _rc!=0 { replace `Ghsqrwt'=0.9*`Ghsqrwt' }
else { local flag=10}
}
replace `Ghsqrwt'=($MA_FBSC)*`Ghsqrwt'
local flag2=0
local flag3=0
local i=1
gph pen 2
while `i'<=`ymax' {
local Aytcen= `r5'*(`id'[`i']-0.2)+`r6' /* text label centre */
local Aygcen= `r5'*(`id'[`i'])+`r6' /* graphics label centre */
*label to put on left hand
local tx = `label'[`i']
local Axtexs=`Axtexh'+400
*use=0 => blank line except for "by" legend
if (`use'[`i']==0) {
gph pen 3
gph text `Aytcen' `Axtexh' 0 -1 `tx'
}
*use=1 => individual trial
if `use'[`i']==1 {
gph pen 2
gph text `Aytcen' `Axtexs' 0 -1 `tx'
if `lci'[`i']==. | `uci'[`i']==. { local flag2=10 }
else {
* Define lower/upper points on x-line, and centre on y-axis
local Axlpt= `r7'*(`xlog'( `lci'[`i'] ))+`r8'
local Axupt= `r7'*(`xlog'( `uci'[`i'] ))+`r8'
if (`Axupt' < `Axlo') | (`Axlpt' > `Axhi') {
* If CI is totally off scale draw (triangular) arrow
local Ayco1=`r5'*(`id'[`i']-0.2)+`r6'
local Ayco2=`r5'*(`id'[`i']+0.2)+`r6'
if `Axupt'<=`Axlo' {
local Axlpt =`Axlo'
local Axco1=`r7'*(`Gxlo')+`r8'
local Axco2=`r7'*(`Gxlo')+`r8'+450
}
if `Axlpt'>=`Axhi' {
local Axupt =`Axhi'
local Axco1=`r7'*(`Gxhi')+`r8'
local Axco2=`r7'*(`Gxhi')+`r8'-450
}
gph line `Aygcen' `Axco1' `Ayco2' `Axco2'
gph line `Ayco2' `Axco2' `Ayco1' `Axco2'
gph line `Ayco1' `Axco2' `Aygcen' `Axco1'
}
else {
local Axcen =`r7'*`xlog'(`effect'[`i'])+`r8'
* Define box size
local Ahboxl =abs(`r5'*( `Ghsqrwt'[`i'] ))
local Ay1cord=`Aygcen'+`Ahboxl'
local Ax1cord=`Axcen' -`Ahboxl'
local Ay2cord=`Aygcen'-`Ahboxl'
local Ax2cord=`Axcen' +`Ahboxl'
if (`Axlpt' < `Axlo') | (`Axupt' > `Axhi') {
* CI is on but not totaly on scale: draw arrow at end of CI
local Ayco1=`r5'*(`id'[`i']-0.1)+`r6'
local Ayco2=`r5'*(`id'[`i']+0.1)+`r6'
if `Axlpt' < `Axlo' {
local Axlpt =`Axlo'
local Axco1=`r7'*(`Gxlo')+`r8'
local Axco2=`r7'*(`Gxlo')+`r8'+350
gph line `Aygcen' `Axco1' `Ayco1' `Axco2'
gph line `Aygcen' `Axco1' `Ayco2' `Axco2'
}
if `Axupt' > `Axhi' {
local Axupt =`Axhi'
local Axco1=`r7'*(`Gxhi')+`r8'
local Axco2=`r7'*(`Gxhi')+`r8'-350
gph line `Aygcen' `Axco1' `Ayco1' `Axco2'
gph line `Aygcen' `Axco1' `Ayco2' `Axco2'
}
}
*draw line for CI
gph line `Aygcen' `Axlpt' `Aygcen' `Axupt'
*either draw box for ES/weight...
if "`box'"=="" {
if (`Ax1cord' >=`Axlo') & (`Ax2cord'<=`Axhi') {
gph box `Ay1cord' `Ax1cord' `Ay2cord' `Ax2cord' $MA_FBSH
}
else {local flag2=10}
}
*...or simply plot ES (as circle) instead of box
if "`box'"!="" {
if (`Ax1cord' >=`Axlo') & (`Ax2cord'<=`Axhi') {
local ptsize=250*$MA_FBSC
gph point `Aygcen' `Axcen' `ptsize' 1
}
else {local flag2=10}
}
}
}
}
*use=2 => Excluded trial
if (`use'[`i']==2) {
gph text `Aytcen' `Axtexs' 0 -1 `tx'
}
*use=4 => blank line: no text or graphic necessary (above displays nothing)
*use=3 => subgroup effect size (display by default), or...
*use=5 => overall effect size (display by default)
if ( ((`use'[`i']==3) & ("`subgrp'"=="")) | /*
*/ ((`use'[`i']==5) & ("`overall'"=="")) ) {
if (`use'[`i']==3) { gph pen 3 }
if (`use'[`i']==5) { gph pen 5 }
gph text `Aytcen' `Axtexh' 0 -1 `tx'
local Aycol=`r5'*((`id'[`i'])-0.2)+`r6'
local Aycoh= `r5'*((`id'[`i'])+0.2)+`r6'
*the following 4 are necc in case diamond is chopped off
local Aycenl1=`Aygcen'
local Aycenl2=`Aygcen'
local Aycenh1=`Aygcen'
local Aycenh2=`Aygcen'
local Axcol=`r7'*(`xlog'(`lci'[`i']))+`r8'
local Axcen=`r7'*(`xlog'(`effect'[`i']))+`r8'
local Axcoh=`r7'*(`xlog'(`uci'[`i']))+`r8'
if (`Axcen'<`Axlo') | (`Axcen'>`Axhi') {
*diamond is off the scale!
local flag3=10
}
else {
* phi is angle between diamond slope and y=id in right angle triangle; use this
* fact to get y where diamond is chopped off at
if `Axcol'<`Axlo' {
local flag3=10
local tanphi=(0.2*`r5')/(`Axcen'-`Axcol')
local Aydiff=(`Axlo'-`Axcol')*`tanphi'
local Aycenl1=`Aygcen'-`Aydiff'
local Aycenl2=`Aygcen'+`Aydiff'
local Axcol=`Axlo'
gph line `Aycenl1' `Axcol' `Aycenl2' `Axcol'
}
if `Axcoh'>`Axhi' {
local flag3=10
local tanphi=(0.2*`r5')/(`Axcoh'-`Axcen')
local Aydiff=(`Axcoh'-`Axhi')*`tanphi'
local Aycenh1=`Aygcen'-`Aydiff'
local Aycenh2=`Aygcen'+`Aydiff'
local Axcoh=`Axhi'
gph line `Aycenh1' `Axcoh' `Aycenh2' `Axcoh'
}
gph line `Aycoh' `Axcen' `Aycenh2' `Axcoh'
gph line `Aycenh1' `Axcoh' `Aycol' `Axcen'
gph line `Aycol' `Axcen' `Aycenl1' `Axcol'
gph line `Aycenl2' `Axcol' `Aycoh' `Axcen'
*Overall line (if specified)
if ((`use'[`i'])==5 & "`line'"=="" ) {
gph pen 5
local Adashl=`r5'*(`Gyhi'-1)/100
local Ayhi =`r5'*`Gyhi'+`r6'
local j =`r5'+`r6'
while `j'>`Ayhi' {
local Aycol=`j'
local Aycoh=`j'+`Adashl'
gph line `Aycol' `Axcen' `Aycoh' `Axcen'
local j=`j'+2*`Adashl'
}
}
}
}
*use=9 => excluded, ignore (will have been sorted to bottom of data)
*Diamonds or boxes&lines are now drawn - put text on the end
if ( `use'[`i']==1 | `use'[`i']==3 | `use'[`i']==5 ) {
*put text at end of graph (if requested)
if "`stats'"=="" {
*effect sizes
local e1=`effect'[`i']
local e2=`lci'[`i']
local e3=`uci'[`i']
*Make allowance for alignment where es<0
local sp1
if (`e1'>0) {local sp1 " "}
local sp2
if (`e2'>0) {local sp2 " "}
local sp3
if (`e3'>0) {local sp3 " "}
if (`e1'<1e-8) & "`Glog'"!="" {local e1 "<10^-8"}
else if (`e1'>1e8) & (`e1'!=.) & "`Glog'"!="" {
local e1 ">10^8"
}
else { local e1: displ %4.2f `e1' }
if (`e2'<1e-8) & ("`Glog'"!="" | "`cornfield'"!="") {
local e2 "<10^-8"
}
else if (`e2'>1e8) & (`e2'!=.) & "`Glog'"!="" {local e2 ">10^8"}
else { local e2: displ %4.2f `e2' }
if (`e3'<1e-8) & "`Glog'"!="" {local e3 "<10^-8"}
else if (`e3'>1e8) & (`e3'!=.) & ("`Glog'"!="" | "`cornfield'"!="") {
local e3 ">10^8"
}
else { local e3: displ %4.2f `e3' }
local esize "`sp1'`e1' (`sp2'`e2',`sp3'`e3')"
gph text `Aytcen' `Axhi' 0 -1 `esize'
}
if "`wt'"=="" {
local weit: displ %4.1f `wtdisp'[`i']
if `weit'!=. { gph text `Aytcen' `Ax3hi' 0 1 `weit' }
}
if "`counts'"!="" {
local nm: displ `rawdata'[`i']
if "`nm'"!="." {
parse "`nm'" , parse(";")
local nm1 "`1'"
local nm2 "`3'"
gph text `Aytcen' `Ax2hi1' 0 1 `nm1'
gph text `Aytcen' `Ax2hi2' 0 1 `nm2'
}
}
}
if `use'[`i']==2 & "`stats'"=="" {
gph text `Aytcen' `Axhi' 0 -1 (Excluded)
if "`counts'"!="" {
local nm: displ `rawdata'[`i']
if "`nm'"!="." {
parse "`nm'" , parse(";")
local nm1 "`1'"
local nm2 "`3'"
gph text `Aytcen' `Ax2hi1' 0 1 `nm1'
gph text `Aytcen' `Ax2hi2' 0 1 `nm2'
}
}
}
local i=`i'+1
}
gph close
} /* end of qui section*/
#delimit ;
if `flag1'>1 { di in bl _n "Note: invalid xlabel() or xtick(): graph has been rescaled"};
if `flag2'>1 { di in bl _n "Warning: Some trials cannot be
represented graphically (may contain" _n "inestimable effects). Consider using different xlabel()"};
if `flag3'>1 { di in bl _n "Warning: Overall (or subgroup) effect size not fully
represented graphically." _n "Consider using xlabel()"};
#delimit cr
end
exit
Revision history
*Log option added: can display Log(ORs,RRs) on table & graph.
*graph appears when only 1 trial.
*nomenclature for _ES changed when OR or RR selected
*std err of estimate /log est in saved results is renamed and labelled so is clearer
*can handle 2 variables (ES, seES), exponentiated with new eform option
*Max label length for table is now 20chrs
*Breslow Day test for OR heterogeneity added
*Can display n/N for trials, with and without weights/stats.
*Groups can be labelled when using the counts option with group1() and group2()
*User-defined weight option:
* - can't use with fixed or random option
* - metan ... , wgt(wt) == metan ... , fixedi if wt=1/v ie IV weighting
* - however will differ if RE:
* - no tau2 involved in weights ie variance est differs: wgt=1/(vi+t) but SIGMA{1/wgt}=/= SIGMA{wi^2 * var(thetai) }/[SIGMA(wi)]^2
* - pooled ES=SIGMA{wgt*es}/SIGMA{wgt} =/= FE weighted, so heterogeneity not equal
*Can plot the ES alone without the box (nobox)
*Option to change "Study" label to something else
*Display a bit nicer (table formatted 3 rather than up to 7 dec places)
*Removed "Study -" from top; by default is replaced with "Study". Annoying "tick" next to it removed
*Added I^2 (Higgins & Thompson 2003)
*xtick, b2title options added can add "Favours ..." below graph with prefix and asterisk in b1(*I: .... * ....)
*Nointeger: allows sample size to be non-integer if requested.
*Adding continuity correction: currently allows user-defined const (or via nointeger opt)
*r(sumstat) now displays with S/WMD (did forget to was long standing bug)
*by() option added!!!
*r(tger) and r(cger) corrected from prev by() versions; overall ERs were not reported (were from the last by subg)
*bug with by() removed: wasn't keeping _ES etc
*allows CI syntax (3 variables: theta lower_limit upper_limit), BUT:
* - does not follow "meta" syntax, ie does not assume log transform needed
* - default is to assume symmetry and calculate se=0.5*(upper-lower)/z, but allows asymmetry
* - bug with udw and missing data fixed 1Mar04
*minor bugs fixed: trial labelling with by() on table, extended line to right of graph with
*options nostats nowt counts all specified, sortby w/o by() option, junk text if by() subgroup
*contained no informative trials, erroneous lack of overall summary on graph in some situations.