import pandas as pd
import rpy2.robjects as ro
from rpy2.robjects import pandas2ri
@@ -248,7 +248,7 @@ Setup
= pf.get_data(model="Feols", N=10_000, seed=99292) data
Setup
Ordinary Least
IID Inference
First, we estimate a model via `pyfixest. We compute “iid” standard errors.
-
+
= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid") fit
We estimate the same model with weights:
-
+
= pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, weights="weights", vcov="iid"
fml )
Via r-fixest
and rpy2
, we get
-
+
= fixest.feols(
r_fit "Y ~ X1 + X2 | f1 + f2"),
ro.Formula(=data,
@@ -357,7 +357,7 @@ dataIID Inference
R[write to console]: NOTE: 3 observations removed because of NA values (LHS: 1, RHS: 1, Fixed-effects: 1).
Let’s compare how close the covariance matrices are:
-
+
= fit._vcov
fit_vcov = stats.vcov(r_fit)
r_vcov - r_vcov fit_vcov
@@ -367,7 +367,7 @@ IID Inference
And for WLS:
-
+
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[ 1.68051337e-18, -2.11758237e-21],
@@ -375,7 +375,7 @@ IID Inference
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit.tidy()
@@ -427,7 +427,7 @@ IID Inference
-
+
pd.DataFrame(broom.tidy_fixest(r_fit)).T
@@ -467,7 +467,7 @@ IID Inference
-
+
fit_weights.tidy()
@@ -519,7 +519,7 @@ IID Inference
-
+
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
@@ -563,13 +563,13 @@ IID Inference
Heteroskedastic Errors
We repeat the same exercise with heteroskedastic (HC1) errors:
-
+
= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero")
fit = pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", weights="weights"
fml )
-
+
= fixest.feols(
r_fit "Y ~ X1 + X2 | f1 + f2"),
ro.Formula(=data,
@@ -592,14 +592,14 @@ dataHeteroskedastic Err
As before, we compare the variance covariance matrices:
-
+
- stats.vcov(r_fit) fit._vcov
array([[-1.61762964e-16, -2.13305660e-17],
[-2.13306190e-17, -5.39492225e-17]])
-
+
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[-2.05022631e-16, -9.53695571e-18],
@@ -607,7 +607,7 @@ Heteroskedastic Err
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit.tidy()
@@ -659,7 +659,7 @@ Heteroskedastic Err
-
+
pd.DataFrame(broom.tidy_fixest(r_fit)).T
@@ -699,7 +699,7 @@ Heteroskedastic Err
-
+
fit_weights.tidy()
@@ -751,7 +751,7 @@ Heteroskedastic Err
-
+
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
@@ -795,7 +795,7 @@ Heteroskedastic Err
Cluster-Robust Errors
We conclude with cluster robust errors.
-
+
= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"})
fit = pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"}, weights="weights"
@@ -821,14 +821,14 @@ fmlCluster-Robust Error
-
+
- stats.vcov(r_fit) fit._vcov
array([[ 4.20670443e-16, -6.97565513e-17],
[-6.97565513e-17, -1.42166010e-17]])
-
+
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[2.59070109e-16, 4.07324592e-16],
@@ -836,7 +836,7 @@ Cluster-Robust Error
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit.tidy()
@@ -888,7 +888,7 @@ Cluster-Robust Error
-
+
pd.DataFrame(broom.tidy_fixest(r_fit)).T
@@ -928,7 +928,7 @@ Cluster-Robust Error
-
+
fit_weights.tidy()
@@ -980,7 +980,7 @@ Cluster-Robust Error
-
+
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
@@ -1024,10 +1024,10 @@ Cluster-Robust Error
Poisson Regression
-
+
= pf.get_data(model="Fepois") data
-
+
= pf.fepois(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid", iwls_tol=1e-10)
fit_iid = pf.fepois(
fit_hetero ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", iwls_tol=1e-10
@@ -1065,21 +1065,21 @@ fmlPoisson Regression
-
+
- stats.vcov(fit_r_iid) fit_iid._vcov
array([[ 1.20791284e-08, -6.55604931e-10],
[-6.55604931e-10, 1.69958097e-09]])
-
+
- stats.vcov(fit_r_hetero) fit_hetero._vcov
array([[ 2.18101847e-08, -7.38711972e-10],
[-7.38711972e-10, 3.07587753e-09]])
-
+
- stats.vcov(fit_r_crv) fit_crv._vcov
array([[ 1.58300904e-08, -1.20806815e-10],
@@ -1087,7 +1087,7 @@ Poisson Regression
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit_iid.tidy()
@@ -1139,7 +1139,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_iid)).T
@@ -1179,7 +1179,7 @@ Poisson Regression
-
+
fit_hetero.tidy()
@@ -1231,7 +1231,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_hetero)).T
@@ -1271,7 +1271,7 @@ Poisson Regression
-
+
fit_crv.tidy()
@@ -1323,7 +1323,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_crv)).T
diff --git a/difference-in-differences.html b/difference-in-differences.html
index 9fdf3693..e42777c0 100644
--- a/difference-in-differences.html
+++ b/difference-in-differences.html
@@ -257,7 +257,7 @@ Difference-in-Differences Estimation
See also NBER SI methods lectures on Linear Panel Event Studies.
Setup
-
+
from importlib import resources
import pandas as pd
@@ -272,7 +272,7 @@ Setup
%autoreload 2
-
+
@@ -306,7 +306,7 @@ Setup
-
+
@@ -344,7 +344,7 @@ Setup
-
+
# one-shot adoption data - parallel trends is true
= get_sharkfin()
df_one_cohort df_one_cohort.head()
@@ -410,7 +410,7 @@ Setup
-
+
# multi-cohort adoption data
= pd.read_csv(
df_multi_cohort "pyfixest.did.data").joinpath("df_het.csv")
@@ -536,7 +536,7 @@ resources.files(Setup
Examining Treatment Timing
Before any DiD estimation, we need to examine the treatment timing, since it is crucial to our choice of estimator.
-
+
pf.panelview(
df_one_cohort,="unit",
@@ -557,7 +557,7 @@ unitExamining Treat
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -580,7 +580,7 @@ unitExamining Treat
We immediately see that we have staggered adoption of treatment in the second case, which implies that a naive application of 2WFE might yield biased estimates under substantial effect heterogeneity.
We can also plot treatment assignment in a disaggregated fashion, which gives us a sense of cohort sizes.
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -604,7 +604,7 @@ unitExamining Treat
Inspecting the Outcome Variable
pf.panelview()
further allows us to inspect the “outcome” variable over time:
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -612,20 +612,19 @@ outcomeInspecting
="year",
time="treat",
treat=True,
- collapse_to_cohort=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
We immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.
We can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -633,13 +632,12 @@ outcomeInspecting
="year",
time="treat",
treat=100,
- subsamp=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
@@ -648,7 +646,7 @@ Inspecting
One-shot adoption: Static and Dynamic Specifications
After taking a first look at the data, let’s turn to estimation. We return to the df_one_cohort
data set (without staggered treatment rollout).
-
+
= pf.feols(
fit_static_twfe "Y ~ treat | unit + year",
@@ -671,14 +669,14 @@ df_one_cohort,
+
= pf.feols(
fit_dynamic_twfe "Y ~ i(year, ever_treated, ref = 14) | unit + year",
df_one_cohort,={"CRV1": "unit"},
vcov )
-
+
fit_dynamic_twfe.iplot(=False,
coord_flip="Event Study",
@@ -688,7 +686,7 @@ title=rename_event_study_coefs(fit_dynamic_twfe._coefnames),
)
labels
-
+
-
+
fit_lpdid.iplot(=False,
coord_flip="Local-Projections-Estimator",
@@ -1167,7 +1165,7 @@ titleLocal Project
=18.5,
xintercept ).show()
-
+
@@ -297,7 +297,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
-
+
@@ -390,7 +390,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
Suppose we were interested in testing the hypothesis that \(X_{1} = X_{2}\). Given the relatively large differences in coefficients and small standard errors, we will likely reject the null that the two parameters are equal.
We can run the formal test via the hypotheses
function from the marginaleffects
package.
-
+
"X1 - X2 = 0") hypotheses(fit,
@@ -545,7 +545,7 @@ PyFixest 0.18.0
Additionally, model_matrix_fixest
now returns a dictionary instead of a tuple.
Brings back fixed effects reference setting via i(var1, var2, ref)
syntax. Deprecates the i_ref1
, i_ref2
function arguments. I.e. it is again possible to e.g. run
-
+
import pyfixest as pf
= pf.get_data()
data
@@ -553,7 +553,7 @@ PyFixest 0.18.0
0:8] fit1.coef()[
Via the ref
syntax, via can set the reference level:
-
+
= pf.feols("Y ~ i(f1, X2, ref = 1)", data=data)
fit2 0:8] fit2.coef()[
@@ -562,7 +562,7 @@ PyFixest 0.18.0
PyFixest 0.17.0
Restructures the codebase and reorganizes how users can interact with the pyfixest
API. It is now recommended to use pyfixest
in the following way:
-
+
import numpy as np
import pyfixest as pf
= pf.get_data()
@@ -630,7 +630,7 @@ data PyFixest 0.17.0
The update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!
Adds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!
-
+
= True) fit.confint(joint
@@ -647,18 +647,18 @@ PyFixest 0.17.0
Intercept
-0.379125
-1.178573
+0.373326
+1.184372
D
--1.759996
--1.045238
+-1.765180
+-1.040053
f1
--0.014143
-0.023692
+-0.014418
+0.023966
@@ -667,7 +667,7 @@ PyFixest 0.17.0
Adds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv()
method.
-
+
= "D", cluster = "group_id") fit.ccv(treatment
/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.
@@ -693,11 +693,11 @@ PyFixest 0.17.0
CCV
-1.4026168622179929
-0.301475
--4.652514
-0.000198
--2.035992
--0.769241
+0.275208
+-5.096563
+0.000075
+-1.980808
+-0.824425
CRV1
@@ -739,7 +739,7 @@ PyFixest 0.14.0
- Changes all docstrings to
numpy
format.
- Difference-in-differences estimation functions now need to be imported via the
pyfixest.did.estimation
module:
-
+
from pyfixest.did.estimation import did2s, lpdid, event_study
diff --git a/quarto_example/QuartoExample.pdf b/quarto_example/QuartoExample.pdf
index 78921f36..41692775 100644
Binary files a/quarto_example/QuartoExample.pdf and b/quarto_example/QuartoExample.pdf differ
diff --git a/quickstart.html b/quickstart.html
index 5bd8b2d2..7fca5546 100644
--- a/quickstart.html
+++ b/quickstart.html
@@ -281,7 +281,7 @@ What is a fix
Read Sample Data
In a first step, we load the module and some synthetic example data:
-
+
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
@@ -302,7 +302,7 @@ Read Sample Data
data.head()
-
+
@@ -336,7 +336,7 @@ Read Sample Data
-
+
@@ -370,7 +370,7 @@ Read Sample Data
-
+
-pandas : 2.2.3
-matplotlib: 3.9.2
-pyfixest : 0.24.3
+pyfixest : 0.24.3
numpy : 1.26.4
+matplotlib: 3.9.2
+pandas : 2.2.3
@@ -507,7 +507,7 @@ Read Sample Data
-
+
data.info()
<class 'pandas.core.frame.DataFrame'>
@@ -535,7 +535,7 @@ Read Sample Data
OLS Estimation
We are interested in the relation between the dependent variable Y
and the independent variables X1
using a fixed effect model for group_id
. Let’s see how the data looks like:
-
+
= data.plot(kind="scatter", x="X1", y="Y", c="group_id", colormap="viridis") ax
@@ -546,7 +546,7 @@ OLS Estimation
We can estimate a fixed effects regression via the feols()
function. feols()
has three arguments: a two-sided model formula, the data, and optionally, the type of inference.
-
+
= pf.feols(fml="Y ~ X1 | group_id", data=data, vcov="HC1")
fit type(fit)
@@ -559,7 +559,7 @@ OLS Estimation
Inspecting Model Results
To inspect the results, we can use a summary function or method:
-
+
fit.summary()
###
@@ -577,54 +577,54 @@ Inspecting Model
Or display a formatted regression table:
-
+
pf.etable(fit)
-
+
@@ -687,7 +687,7 @@ Inspecting Model
Alternatively, the .summarize
module contains a summary
function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable()
, please take a look at the dedicated vignette.
-
+
pf.summary(fit)
###
@@ -705,7 +705,7 @@ Inspecting Model
You can access individual elements of the summary via dedicated methods: .tidy()
returns a “tidy” pd.DataFrame
, .coef()
returns estimated parameters, and se()
estimated standard errors. Other methods include pvalue()
, confint()
and tstat()
.
-
+
fit.tidy()
@@ -748,7 +748,7 @@ Inspecting Model
-
+
fit.coef()
Coefficient
@@ -756,7 +756,7 @@ Inspecting Model
Name: Estimate, dtype: float64
-
+
fit.se()
Coefficient
@@ -764,7 +764,7 @@ Inspecting Model
Name: Std. Error, dtype: float64
-
+
fit.tstat()
Coefficient
@@ -772,7 +772,7 @@ Inspecting Model
Name: t value, dtype: float64
-
+
fit.confint()
@@ -799,11 +799,11 @@ Inspecting Model
Last, model results can be visualized via dedicated methods for plotting:
-
+
fit.coefplot()# or pf.coefplot([fit])
-
+
@@ -522,7 +522,7 @@ Examples
-
+
@@ -671,7 +671,7 @@ Examples
In a first step, we estimate a classical event study model:
-
+
# estimate the model
= pf.did2s(
fit
@@ -761,10 +761,10 @@ df_het,Examples
We can also inspect the model visually:
-
+
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -545,7 +545,7 @@ Examples
-
+
diff --git a/reference/did.estimation.lpdid.html b/reference/did.estimation.lpdid.html
index 6d7187ce..b74e15d2 100644
--- a/reference/did.estimation.lpdid.html
+++ b/reference/did.estimation.lpdid.html
@@ -505,7 +505,7 @@ Returns
Examples
-
+
import pandas as pd
import pyfixest as pf
@@ -528,7 +528,7 @@ Examples
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -562,7 +562,7 @@ Examples
-
+
-
+
@@ -588,7 +588,7 @@ Examples
-
+
@@ -638,7 +638,7 @@ Examples
Calling feols()
returns an instance of the [Feols(/reference/Feols.qmd) class. The summary()
method can be used to print the results.
An alternative way to retrieve model results is via the tidy()
method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.
-
+
fit.tidy()
@@ -692,17 +692,17 @@ Examples
You can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef()
for the coefficients, fit.se()
for the standard errors, fit.tstat()
for the t-statistics, and fit.pval()
for the p-values, and fit.confint()
for the confidence intervals.
The employed type of inference can be specified via the vcov
argument. If vcov is not provided, PyFixest
employs the fixest
default of iid inference, unless there are fixed effects in the model, in which case feols()
clusters the standard error by the first fixed effect (CRV1 inference).
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="iid")
fit1 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="hetero")
fit2 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1"}) fit3
Supported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {"CRV1": "f1"}
for CRV1 inference with clustering by f1
or {"CRV3": "f1"}
for CRV3 inference with clustering by f1
. For two-way clustering, you can provide a formula string, e.g. {"CRV1": "f1 + f2"}
for CRV1 inference with clustering by f1
.
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1 + f2"}) fit4
Inference can be adjusted post estimation via the vcov
method:
-
+
fit.summary()"iid").summary() fit.vcov(
@@ -736,7 +736,7 @@ Examples
The ssc
argument specifies the small sample correction for inference. In general, feols()
uses all of fixest::feols()
defaults, but sets the fixef.K
argument to "none"
whereas the fixest::feols()
default is "nested"
. See here for more details: link to github.
feols()
supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1
and one with fixed effects for f2
using the sw()
syntax.
-
+
= pf.feols("Y ~ X1 + X2 | sw(f1, f2)", data)
fit type(fit)
@@ -744,58 +744,58 @@ Examples
The returned object is an instance of the FixestMulti
class. You can access the results of the first model via fit.fetch_model(0)
and the results of the second model via fit.fetch_model(1)
. You can compare the model results via the etable()
function:
-
+
0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
Model: Y~X1+X2|f1
Model: Y~X1+X2|f2
-
+
@@ -837,14 +837,14 @@ Examples
fe
-f1
-x
+f2
-
+x
-f2
--
+f1
x
+-
stats
@@ -878,7 +878,7 @@ Examples
Other supported multiple estimation syntax include sw0()
, csw()
and csw0()
. While sw()
adds variables in a “stepwise” fashion, csw()
does so cumulatively.
-
+
= pf.feols("Y ~ X1 + X2 | csw(f1, f2)", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -886,51 +886,51 @@ Examples
Model: Y~X1+X2|f1+f2
-
+
@@ -972,13 +972,13 @@ Examples
fe
-f1
-x
+f2
+-
x
-f2
--
+f1
+x
x
@@ -1013,7 +1013,7 @@ Examples
The sw0()
and csw0()
syntax are similar to sw()
and csw()
, but start with a model that excludes the variables specified in sw()
and csw()
:
-
+
= pf.feols("Y ~ X1 + X2 | sw0(f1, f2)", data)
fit 0), fit.fetch_model(1), fit.fetch_model(2)]) pf.etable([fit.fetch_model(
@@ -1022,51 +1022,51 @@ Examples
Model: Y~X1+X2|f2
-
+
@@ -1121,16 +1121,16 @@ Examples
fe
-f1
+f2
-
-x
-
+x
-f2
--
+f1
-
x
+-
stats
@@ -1167,7 +1167,7 @@ Examples
The feols()
function also supports multiple dependent variables. The following example estimates two models, one with Y1
as the dependent variable and one with Y2
as the dependent variable.
-
+
= pf.feols("Y + Y2 ~ X1 | f1 + f2", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -1175,51 +1175,51 @@ Examples
Model: Y2~X1|f1+f2
-
+
@@ -1255,12 +1255,12 @@ Examples
fe
-f1
+f2
x
x
-f2
+f1
x
x
@@ -1296,7 +1296,7 @@ Examples
It is possible to combine different multiple estimation operators:
-
+
= pf.feols("Y + Y2 ~ X1 | sw(f1, f2)", data)
fit 0),
pf.etable([fit.fetch_model(1),
@@ -1311,51 +1311,51 @@ fit.fetch_model(Examples
Model: Y2~X1|f2
-
+
@@ -1401,18 +1401,18 @@ Examples
fe
-f1
-x
-x
+f2
-
-
+x
+x
-f2
--
--
+f1
x
x
+-
+-
stats
@@ -1453,7 +1453,7 @@ Examples
In general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols()
implements a caching mechanism that stores the demeaned covariates.
Besides OLS, feols()
also supports IV estimation via three part formulas:
-
+
= pf.feols("Y ~ X2 | f1 + f2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1507,7 +1507,7 @@ Examples
Here, X1
is the endogenous variable and Z1
is the instrument. f1
and f2
are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:
-
+
= pf.feols("Y ~ X2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1571,7 +1571,7 @@ Examples
Last, feols()
supports interaction of variables via the i()
syntax. Documentation on this is tba.
After fitting a model via feols()
, you can use the predict()
method to get the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit 0:5] fit.predict()[
@@ -1579,7 +1579,7 @@ Examples
The predict()
method also supports a newdata
argument to predict on new data, which returns a numpy array of the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit =data)[0:5] fit.predict(newdata
@@ -1587,11 +1587,11 @@ Examples
Last, you can plot the results of a model via the coefplot()
method:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit fit.coefplot()
-
+
@@ -569,7 +569,7 @@ Examples
-
+
diff --git a/reference/report.coefplot.html b/reference/report.coefplot.html
index d8796924..098ac83c 100644
--- a/reference/report.coefplot.html
+++ b/reference/report.coefplot.html
@@ -528,7 +528,7 @@ Returns
Examples
-
+
import pyfixest as pf
from pyfixest.report.utils import rename_categoricals
@@ -544,7 +544,7 @@ Examples
= "both") pf.coefplot([fit1], joint
-
+
@@ -578,7 +578,7 @@ Examples
-
+
-
+
@@ -576,7 +576,7 @@ Examples
-
+
-
+
@@ -497,7 +497,7 @@ Examples
-
+
diff --git a/replicating-the-effect.html b/replicating-the-effect.html
index 804007ca..a6a6a321 100644
--- a/replicating-the-effect.html
+++ b/replicating-the-effect.html
@@ -234,7 +234,7 @@ Replicating Examples from “The Effect”
This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.
-
+
from causaldata import Mroz, gapminder, organ_donations, restaurant_inspections
import pyfixest as pf
@@ -243,7 +243,7 @@ Replicating Examples from “The Effect”
%watermark --iversions
-
+
@@ -277,7 +277,7 @@ Replicating Examples from “The Effect”
-
+
-causaldata: 0.1.4
-pyfixest : 0.24.3
+pyfixest : 0.24.3
+causaldata: 0.1.4
Chapter 4: Describing Relationships
-
+
# Read in data
= Mroz.load_pandas().data
dt # Keep just working women
@@ -329,7 +329,7 @@ Chapter
= pf.feols(fml="lwg ~ csw(inc, wc, k5)", data=dt, vcov="iid")
fit pf.etable(fit)
-/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning:
+/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning:
A value is trying to be set on a copy of a slice from a DataFrame.
Try using .loc[row_indexer,col_indexer] = value instead
@@ -337,51 +337,51 @@ Chapter
dt.loc[:, "earn"] = dt["lwg"].apply("exp")
-
+
@@ -479,7 +479,7 @@ Chapter
Chapter 13: Regression
Example 1
-
+
= restaurant_inspections.load_pandas().data
res = res.inspection_score.astype(float)
res.inspection_score = res.NumberofLocations.astype(float)
@@ -488,51 +488,51 @@ res.NumberofLocations Example 1
= pf.feols(fml="inspection_score ~ NumberofLocations", data=res)
fit pf.etable([fit])
-
+
@@ -595,7 +595,7 @@ Example 1
Example 2
-
+
= restaurant_inspections.load_pandas().data
df
= pf.feols(
@@ -605,51 +605,51 @@ fit1 Example 2
pf.etable([fit1, fit2])
-
+
@@ -746,7 +746,7 @@ Example 2
Example 3: HC Standard Errors
-
+
="inspection_score ~ Year + Weekend", data=df, vcov="HC3").summary() pf.feols(fml
###
@@ -768,7 +768,7 @@ Example 3: HC
Example 4: Clustered Standard Errors
-
+
pf.feols(="inspection_score ~ Year + Weekend", data=df, vcov={"CRV1": "Weekend"}
fml ).tidy()
@@ -834,7 +834,7 @@ Exampl
Example 5: Bootstrap Inference
-
+
= pf.feols(fml="inspection_score ~ Year + Weekend", data=df)
fit =999, param="Year") fit.wildboottest(reps
@@ -857,7 +857,7 @@ Example 1
Example 2
-
+
= gapminder.load_pandas().data
gm "logGDPpercap"] = gm["gdpPercap"].apply("log")
gm[
@@ -943,7 +943,7 @@ Example 2
Example 3: TWFE
-
+
# Set our individual and time (index) for our data
= pf.feols(fml="lifeExp ~ np.log(gdpPercap) | country + year", data=gm)
fit fit.summary()
@@ -968,7 +968,7 @@ Example 3: TWFE
Chapter 18: Difference-in-Differences
Example 1
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
@@ -996,7 +996,7 @@ Example 1
Example 3: Dynamic Treatment Effect
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
diff --git a/search.json b/search.json
index f717fbdc..f69e7216 100644
--- a/search.json
+++ b/search.json
@@ -74,7 +74,7 @@
"href": "news.html#pyfixest-0.17.0",
"title": "News",
"section": "PyFixest 0.17.0",
- "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.379125\n1.178573\n\n\nD\n-1.759996\n-1.045238\n\n\nf1\n-0.014143\n0.023692\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.301475\n-4.652514\n0.000198\n-2.035992\n-0.769241\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
+ "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.373326\n1.184372\n\n\nD\n-1.765180\n-1.040053\n\n\nf1\n-0.014418\n0.023966\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.275208\n-5.096563\n0.000075\n-1.980808\n-0.824425\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
},
{
"objectID": "news.html#pyfixest-0.16.0",
@@ -298,7 +298,7 @@
"href": "quickstart.html",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#what-is-a-fixed-effect-model",
@@ -312,7 +312,7 @@
"href": "quickstart.html#read-sample-data",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
+ "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
},
{
"objectID": "quickstart.html#ols-estimation",
@@ -340,7 +340,7 @@
"href": "quickstart.html#updating-regression-coefficients",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#inference-via-the-wild-bootstrap",
@@ -375,7 +375,7 @@
"href": "quickstart.html#joint-confidence-intervals",
"title": "Getting Started with PyFixest",
"section": "Joint Confidence Intervals",
- "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.425936\n1.403847\n\n\nX1\n-1.160730\n-0.738152\n\n\nC(f1)[T.1.0]\n1.384234\n3.781064\n\n\nC(f1)[T.2.0]\n-2.838865\n-0.325003\n\n\nC(f1)[T.3.0]\n-1.608332\n0.983664"
+ "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.430485\n1.408396\n\n\nX1\n-1.161781\n-0.737102\n\n\nC(f1)[T.1.0]\n1.378276\n3.787022\n\n\nC(f1)[T.2.0]\n-2.845114\n-0.318754\n\n\nC(f1)[T.3.0]\n-1.614775\n0.990107"
},
{
"objectID": "contributing.html",
@@ -911,7 +911,7 @@
"href": "reference/estimation.estimation.feols.html#examples",
"title": "estimation.estimation.feols",
"section": "Examples",
- "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf1\nx\n-\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf1\n-\nx\n-\n\n\nf2\n-\n-\nx\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf1\nx\nx\n-\n-\n\n\nf2\n-\n-\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.265821\n0.060521\n0.952408\n-0.542381\n0.574556\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
+ "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\n-\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf2\n-\n-\nx\n\n\nf1\n-\nx\n-\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf2\nx\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf2\n-\n-\nx\nx\n\n\nf1\nx\nx\n-\n-\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.25692\n0.062617\n0.950761\n-0.523682\n0.555857\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
"crumbs": [
"Function Reference",
"Estimation Functions",
@@ -1435,14 +1435,14 @@
"href": "table-layout.html#basic-usage",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage",
- "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
+ "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
},
{
"objectID": "table-layout.html#keep-and-drop-variables",
"href": "table-layout.html#keep-and-drop-variables",
"title": "Regression Tables via pf.etable()",
"section": "Keep and drop variables",
- "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#hide-fixed-effects-or-se-type-rows",
@@ -1456,49 +1456,49 @@
"href": "table-layout.html#display-p-values-or-confidence-intervals",
"title": "Regression Tables via pf.etable()",
"section": "Display p-values or confidence intervals",
- "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#significance-levels-and-rounding",
"href": "table-layout.html#significance-levels-and-rounding",
"title": "Regression Tables via pf.etable()",
"section": "Significance levels and rounding",
- "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
+ "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
},
{
"objectID": "table-layout.html#other-output-formats",
"href": "table-layout.html#other-output-formats",
"title": "Regression Tables via pf.etable()",
"section": "Other output formats",
- "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf2 - x x - x x\nf1 x x x x x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
+ "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf1 x x x x x x\nf2 - x x - x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
},
{
"objectID": "table-layout.html#rename-variables",
"href": "table-layout.html#rename-variables",
"title": "Regression Tables via pf.etable()",
"section": "Rename variables",
- "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-model-headlines",
"href": "table-layout.html#custom-model-headlines",
"title": "Regression Tables via pf.etable()",
"section": "Custom model headlines",
- "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
+ "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
},
{
"objectID": "table-layout.html#further-custom-model-information",
"href": "table-layout.html#further-custom-model-information",
"title": "Regression Tables via pf.etable()",
"section": "Further custom model information",
- "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-table-notes",
"href": "table-layout.html#custom-table-notes",
"title": "Regression Tables via pf.etable()",
"section": "Custom table notes",
- "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
+ "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
},
{
"objectID": "table-layout.html#publication-ready-latex-tables",
@@ -1519,49 +1519,49 @@
"href": "table-layout.html#summarize-by-characteristics-in-columns-and-rows",
"title": "Regression Tables via pf.etable()",
"section": "Summarize by characteristics in columns and rows",
- "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.19\n 2.35\n -0.04\n 2.21\n -0.05\n 2.28\n -0.23\n 2.39\n \n \n Wealth\n -0.72\n 5.71\n 0.05\n 4.99\n -0.09\n 5.53\n -0.56\n 6.06\n \n \n Age\n 1.07\n 0.81\n 0.96\n 0.80\n 1.08\n 0.81\n 1.06\n 0.81\n \n \n Years of Schooling\n 0.05\n 3.10\n -0.28\n 2.86\n -0.18\n 3.13\n -0.09\n 3.08\n \n \n nobs\n \n \n Number of observations\n 229\n \n 244\n \n 270\n \n 254\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.19(2.35)\n -0.04(2.21)\n -0.05(2.28)\n -0.23(2.39)\n \n \n Wealth\n -0.72(5.71)\n 0.05(4.99)\n -0.09(5.53)\n -0.56(6.06)\n \n \n Age\n 1.07(0.81)\n 0.96(0.80)\n 1.08(0.81)\n 1.06(0.81)\n \n \n Years of Schooling\n 0.05(3.10)\n -0.28(2.86)\n -0.18(3.13)\n -0.09(3.08)\n \n \n nobs\n \n \n Number of observations\n 229\n 244\n 270\n 254\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 229\n -0.19\n 2.35\n 270\n -0.05\n 2.28\n \n \n Wealth\n 229\n -0.72\n 5.71\n 270\n -0.09\n 5.53\n \n \n Age\n 229\n 1.07\n 0.81\n 270\n 1.08\n 0.81\n \n \n Years of Schooling\n 229\n 0.05\n 3.10\n 270\n -0.18\n 3.13\n \n \n White collar\n \n \n Wage\n 244\n -0.04\n 2.21\n 254\n -0.23\n 2.39\n \n \n Wealth\n 244\n 0.05\n 4.99\n 254\n -0.56\n 6.06\n \n \n Age\n 244\n 0.96\n 0.80\n 254\n 1.06\n 0.81\n \n \n Years of Schooling\n 244\n -0.28\n 2.86\n 254\n -0.09\n 3.08\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
+ "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.18\n 2.35\n 0.10\n 2.26\n -0.28\n 2.33\n -0.13\n 2.28\n \n \n Wealth\n -0.33\n 5.51\n -0.10\n 5.54\n -0.39\n 5.48\n -0.45\n 5.83\n \n \n Age\n 1.05\n 0.82\n 1.05\n 0.81\n 1.02\n 0.81\n 1.05\n 0.80\n \n \n Years of Schooling\n -0.14\n 2.99\n -0.31\n 2.92\n 0.12\n 3.24\n -0.19\n 3.02\n \n \n nobs\n \n \n Number of observations\n 254\n \n 242\n \n 246\n \n 255\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.18(2.35)\n 0.10(2.26)\n -0.28(2.33)\n -0.13(2.28)\n \n \n Wealth\n -0.33(5.51)\n -0.10(5.54)\n -0.39(5.48)\n -0.45(5.83)\n \n \n Age\n 1.05(0.82)\n 1.05(0.81)\n 1.02(0.81)\n 1.05(0.80)\n \n \n Years of Schooling\n -0.14(2.99)\n -0.31(2.92)\n 0.12(3.24)\n -0.19(3.02)\n \n \n nobs\n \n \n Number of observations\n 254\n 242\n 246\n 255\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 254\n -0.18\n 2.35\n 246\n -0.28\n 2.33\n \n \n Wealth\n 254\n -0.33\n 5.51\n 246\n -0.39\n 5.48\n \n \n Age\n 254\n 1.05\n 0.82\n 246\n 1.02\n 0.81\n \n \n Years of Schooling\n 254\n -0.14\n 2.99\n 246\n 0.12\n 3.24\n \n \n White collar\n \n \n Wage\n 242\n 0.10\n 2.26\n 255\n -0.13\n 2.28\n \n \n Wealth\n 242\n -0.10\n 5.54\n 255\n -0.45\n 5.83\n \n \n Age\n 242\n 1.05\n 0.81\n 255\n 1.05\n 0.80\n \n \n Years of Schooling\n 242\n -0.31\n 2.92\n 255\n -0.19\n 3.02\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
},
{
"objectID": "table-layout.html#basic-usage-of-make_table",
"href": "table-layout.html#basic-usage-of-make_table",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage of make_table",
- "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n -0.04\n -1.77\n -1.68\n 1.89\n \n \n 1\n 0.39\n 1.05\n -0.07\n -1.59\n \n \n 2\n 1.1\n -0.18\n 0.3\n -1.25\n \n \n 3\n 1.78\n -0.18\n -0.09\n -0.08\n \n\n \n \n \n These are notes"
+ "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n 0.87\n -1.25\n -1.78\n 1.12\n \n \n 1\n -0.88\n 1.08\n -0.47\n -0.5\n \n \n 2\n -0.31\n 1.04\n 0.56\n 0.12\n \n \n 3\n 0.37\n -0.63\n -0.72\n 1.4\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#mutiindex-dataframes",
"href": "table-layout.html#mutiindex-dataframes",
"title": "Regression Tables via pf.etable()",
"section": "Mutiindex DataFrames",
- "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes"
+ "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#example-styling",
"href": "table-layout.html#example-styling",
"title": "Regression Tables via pf.etable()",
"section": "Example Styling",
- "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#defining-table-styles-some-examples",
"href": "table-layout.html#defining-table-styles-some-examples",
"title": "Regression Tables via pf.etable()",
"section": "Defining Table Styles: Some Examples",
- "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "replicating-the-effect.html",
"href": "replicating-the-effect.html",
"title": "Replicating Examples from “The Effect”",
"section": "",
- "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\ncausaldata: 0.1.4\npyfixest : 0.24.3"
+ "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\ncausaldata: 0.1.4"
},
{
"objectID": "replicating-the-effect.html#chapter-4-describing-relationships",
"href": "replicating-the-effect.html#chapter-4-describing-relationships",
"title": "Replicating Examples from “The Effect”",
"section": "Chapter 4: Describing Relationships",
- "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
+ "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
},
{
"objectID": "replicating-the-effect.html#chapter-13-regression",
@@ -1624,7 +1624,7 @@
"href": "difference-in-differences.html#inspecting-the-outcome-variable",
"title": "Difference-in-Differences Estimation",
"section": "Inspecting the Outcome Variable",
- "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)"
+ "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n title = \"Outcome Plot\"\n)"
},
{
"objectID": "difference-in-differences.html#one-shot-adoption-static-and-dynamic-specifications",
diff --git a/table-layout.html b/table-layout.html
index 3dd0c424..2773792b 100644
--- a/table-layout.html
+++ b/table-layout.html
@@ -245,7 +245,7 @@ Regression Tables via pf.etable()
Table Layout with PyFixest
Pyfixest comes with functions to generate publication-ready tables. Regression tables are generated with pf.etable()
, which can output different formats, for instance using the Great Tables package or generating formatted LaTex Tables using booktabs. There are also further functions pf.dtable()
to display descriptive statistics and pf.make_table()
generating formatted tables from pandas dataframes in the same layout.
To begin, we load some libraries and fit a set of regression models.
-
+
import numpy as np
import pandas as pd
import pylatex as pl # for the latex table; note: not a dependency of pyfixest - needs manual installation
@@ -267,7 +267,7 @@ Table Layout wi
= pf.feols("Y2 ~ X1 *X2 | f1 + f2", data=data) fit6
-
+
@@ -301,7 +301,7 @@ Table Layout wi
-
+
@@ -338,54 +338,54 @@ Table Layout wi
Basic Usage
We can compare all regression models via the pyfixest-internal pf.etable()
function:
-
+
pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
-
+
@@ -444,20 +444,20 @@ Basic Usage
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -507,54 +507,54 @@ Basic Usage
You can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:
-
+
"Y+Y2~csw(X1,X2,X1:X2)", data=data)) pf.etable(pf.feols(
-
+
@@ -667,54 +667,54 @@ Basic Usage
Keep and drop variables
etable
allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.
-
+
="X1") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -764,20 +764,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -827,54 +827,54 @@ Keep and drop vari
We can use the exact_match
argument to select a specific set of variables:
-
+
=["X1", "X2"], exact_match=True) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -924,20 +924,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -987,54 +987,54 @@ Keep and drop vari
We can also easily drop variables via the drop
argument:
-
+
=["X1"]) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop
-
+
@@ -1075,20 +1075,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1141,54 +1141,54 @@ Keep and drop vari
Hide fixed effects or SE-type rows
We can hide the rows showing the relevant fixed effects and those showing the S.E. type by setting show_fe=False
and show_setype=False
(for instance when the set of fixed effects or the estimation method for the std. errors is the same for all models and you want to describe this in the text or table notes rather than displaying it in the table).
-
+
=False, show_se_type=False) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], show_fe
-
+
@@ -1283,54 +1283,54 @@ Hide fi
Display p-values or confidence intervals
By default, pf.etable()
reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt
function argument.
-
+
="b \n (se) \n [p]") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt
-
+
@@ -1389,20 +1389,20 @@ D
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1455,54 +1455,54 @@ D
Significance levels and rounding
Additionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code
and digits
function arguments:
-
+
=[0.01, 0.05, 0.1], digits=5) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code
-
+
@@ -1561,20 +1561,20 @@ Significa
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1627,7 +1627,7 @@ Significa
Other output formats
By default, pf.etable()
returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type
argument.
-
+
# Pandas styler output:
pf.etable(
@@ -1689,20 +1689,20 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
-0.04082 (0.08093)
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1738,7 +1738,7 @@ Other output formats<
-
+
# Markdown output:
pf.etable(
@@ -1758,8 +1758,8 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
X1:X2 0.01057 -0.04082
(0.01818) (0.08093)
------------------------------------------------------------------------------------------------
-f2 - x x - x x
f1 x x x x x x
+f2 - x x - x x
------------------------------------------------------------------------------------------------
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
@@ -1769,7 +1769,7 @@ Other output formats<
To obtain latex output use format = "tex"
. If you want to save the table as a tex file, you can use the filename=
argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True
argument. Etable will use latex packages booktabs
, threeparttable
and makecell
for the table layout, so don’t forget to include these packages in your latex document.
-
+
# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):
= pf.etable(
tab
@@ -1780,7 +1780,7 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
)
The following code generates a pdf including the regression table which you can display clicking on the link below the cell:
-
+
## Use pylatex to create a tex file with the table
@@ -1814,7 +1814,7 @@ Other output formats<
Rename variables
You can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels
argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).
-
+
IID Inference
First, we estimate a model via `pyfixest. We compute “iid” standard errors.
-= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid") fit
We estimate the same model with weights:
-= pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, weights="weights", vcov="iid"
fml )
Via r-fixest
and rpy2
, we get
= fixest.feols(
r_fit "Y ~ X1 + X2 | f1 + f2"),
ro.Formula(=data,
@@ -357,7 +357,7 @@ dataIID Inference
R[write to console]: NOTE: 3 observations removed because of NA values (LHS: 1, RHS: 1, Fixed-effects: 1).
Let’s compare how close the covariance matrices are:
-= fit._vcov
fit_vcov = stats.vcov(r_fit)
r_vcov - r_vcov fit_vcov
IID Inference
And for WLS:
-- stats.vcov(r_fit_weights) fit_weights._vcov
array([[ 1.68051337e-18, -2.11758237e-21],
@@ -375,7 +375,7 @@ IID Inference
We conclude by comparing all estimation results via the tidy
methods:
fit.tidy()
IID Inference
pd.DataFrame(broom.tidy_fixest(r_fit)).T
IID Inference
fit_weights.tidy()
IID Inference
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
IID Inference
Heteroskedastic Errors
We repeat the same exercise with heteroskedastic (HC1) errors:
-= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero")
fit = pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", weights="weights"
fml )
= fixest.feols(
r_fit "Y ~ X1 + X2 | f1 + f2"),
ro.Formula(=data,
@@ -592,14 +592,14 @@ dataHeteroskedastic Err
As before, we compare the variance covariance matrices:
-- stats.vcov(r_fit) fit._vcov
array([[-1.61762964e-16, -2.13305660e-17],
[-2.13306190e-17, -5.39492225e-17]])
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[-2.05022631e-16, -9.53695571e-18],
@@ -607,7 +607,7 @@ Heteroskedastic Err
We conclude by comparing all estimation results via the tidy
methods:
fit.tidy()
Heteroskedastic Err
pd.DataFrame(broom.tidy_fixest(r_fit)).T
Heteroskedastic Err
fit_weights.tidy()
Heteroskedastic Err
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
Heteroskedastic Err
Cluster-Robust Errors
We conclude with cluster robust errors.
-
+
= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"})
fit = pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"}, weights="weights"
@@ -821,14 +821,14 @@ fmlCluster-Robust Error
-
+
- stats.vcov(r_fit) fit._vcov
array([[ 4.20670443e-16, -6.97565513e-17],
[-6.97565513e-17, -1.42166010e-17]])
-
+
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[2.59070109e-16, 4.07324592e-16],
@@ -836,7 +836,7 @@ Cluster-Robust Error
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit.tidy()
@@ -888,7 +888,7 @@ Cluster-Robust Error
-
+
pd.DataFrame(broom.tidy_fixest(r_fit)).T
@@ -928,7 +928,7 @@ Cluster-Robust Error
-
+
fit_weights.tidy()
@@ -980,7 +980,7 @@ Cluster-Robust Error
-
+
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
@@ -1024,10 +1024,10 @@ Cluster-Robust Error
Poisson Regression
-
+
= pf.get_data(model="Fepois") data
-
+
= pf.fepois(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid", iwls_tol=1e-10)
fit_iid = pf.fepois(
fit_hetero ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", iwls_tol=1e-10
@@ -1065,21 +1065,21 @@ fmlPoisson Regression
-
+
- stats.vcov(fit_r_iid) fit_iid._vcov
array([[ 1.20791284e-08, -6.55604931e-10],
[-6.55604931e-10, 1.69958097e-09]])
-
+
- stats.vcov(fit_r_hetero) fit_hetero._vcov
array([[ 2.18101847e-08, -7.38711972e-10],
[-7.38711972e-10, 3.07587753e-09]])
-
+
- stats.vcov(fit_r_crv) fit_crv._vcov
array([[ 1.58300904e-08, -1.20806815e-10],
@@ -1087,7 +1087,7 @@ Poisson Regression
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit_iid.tidy()
@@ -1139,7 +1139,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_iid)).T
@@ -1179,7 +1179,7 @@ Poisson Regression
-
+
fit_hetero.tidy()
@@ -1231,7 +1231,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_hetero)).T
@@ -1271,7 +1271,7 @@ Poisson Regression
-
+
fit_crv.tidy()
@@ -1323,7 +1323,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_crv)).T
diff --git a/difference-in-differences.html b/difference-in-differences.html
index 9fdf3693..e42777c0 100644
--- a/difference-in-differences.html
+++ b/difference-in-differences.html
@@ -257,7 +257,7 @@ Difference-in-Differences Estimation
See also NBER SI methods lectures on Linear Panel Event Studies.
Setup
-
+
from importlib import resources
import pandas as pd
@@ -272,7 +272,7 @@ Setup
%autoreload 2
-
+
@@ -306,7 +306,7 @@ Setup
-
+
@@ -344,7 +344,7 @@ Setup
-
+
# one-shot adoption data - parallel trends is true
= get_sharkfin()
df_one_cohort df_one_cohort.head()
@@ -410,7 +410,7 @@ Setup
-
+
# multi-cohort adoption data
= pd.read_csv(
df_multi_cohort "pyfixest.did.data").joinpath("df_het.csv")
@@ -536,7 +536,7 @@ resources.files(Setup
Examining Treatment Timing
Before any DiD estimation, we need to examine the treatment timing, since it is crucial to our choice of estimator.
-
+
pf.panelview(
df_one_cohort,="unit",
@@ -557,7 +557,7 @@ unitExamining Treat
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -580,7 +580,7 @@ unitExamining Treat
We immediately see that we have staggered adoption of treatment in the second case, which implies that a naive application of 2WFE might yield biased estimates under substantial effect heterogeneity.
We can also plot treatment assignment in a disaggregated fashion, which gives us a sense of cohort sizes.
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -604,7 +604,7 @@ unitExamining Treat
Inspecting the Outcome Variable
pf.panelview()
further allows us to inspect the “outcome” variable over time:
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -612,20 +612,19 @@ outcomeInspecting
="year",
time="treat",
treat=True,
- collapse_to_cohort=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
We immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.
We can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -633,13 +632,12 @@ outcomeInspecting
="year",
time="treat",
treat=100,
- subsamp=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
@@ -648,7 +646,7 @@ Inspecting
One-shot adoption: Static and Dynamic Specifications
After taking a first look at the data, let’s turn to estimation. We return to the df_one_cohort
data set (without staggered treatment rollout).
-
+
= pf.feols(
fit_static_twfe "Y ~ treat | unit + year",
@@ -671,14 +669,14 @@ df_one_cohort,
+
= pf.feols(
fit_dynamic_twfe "Y ~ i(year, ever_treated, ref = 14) | unit + year",
df_one_cohort,={"CRV1": "unit"},
vcov )
-
+
fit_dynamic_twfe.iplot(=False,
coord_flip="Event Study",
@@ -688,7 +686,7 @@ title=rename_event_study_coefs(fit_dynamic_twfe._coefnames),
)
labels
-
+
-
+
fit_lpdid.iplot(=False,
coord_flip="Local-Projections-Estimator",
@@ -1167,7 +1165,7 @@ titleLocal Project
=18.5,
xintercept ).show()
-
+
@@ -297,7 +297,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
-
+
@@ -390,7 +390,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
Suppose we were interested in testing the hypothesis that \(X_{1} = X_{2}\). Given the relatively large differences in coefficients and small standard errors, we will likely reject the null that the two parameters are equal.
We can run the formal test via the hypotheses
function from the marginaleffects
package.
-
+
"X1 - X2 = 0") hypotheses(fit,
@@ -545,7 +545,7 @@ PyFixest 0.18.0
Additionally, model_matrix_fixest
now returns a dictionary instead of a tuple.
Brings back fixed effects reference setting via i(var1, var2, ref)
syntax. Deprecates the i_ref1
, i_ref2
function arguments. I.e. it is again possible to e.g. run
-
+
import pyfixest as pf
= pf.get_data()
data
@@ -553,7 +553,7 @@ PyFixest 0.18.0
0:8] fit1.coef()[
Via the ref
syntax, via can set the reference level:
-
+
= pf.feols("Y ~ i(f1, X2, ref = 1)", data=data)
fit2 0:8] fit2.coef()[
@@ -562,7 +562,7 @@ PyFixest 0.18.0
PyFixest 0.17.0
Restructures the codebase and reorganizes how users can interact with the pyfixest
API. It is now recommended to use pyfixest
in the following way:
-
+
import numpy as np
import pyfixest as pf
= pf.get_data()
@@ -630,7 +630,7 @@ data PyFixest 0.17.0
The update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!
Adds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!
-
+
= True) fit.confint(joint
@@ -647,18 +647,18 @@ PyFixest 0.17.0
Intercept
-0.379125
-1.178573
+0.373326
+1.184372
D
--1.759996
--1.045238
+-1.765180
+-1.040053
f1
--0.014143
-0.023692
+-0.014418
+0.023966
@@ -667,7 +667,7 @@ PyFixest 0.17.0
Adds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv()
method.
-
+
= "D", cluster = "group_id") fit.ccv(treatment
/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.
@@ -693,11 +693,11 @@ PyFixest 0.17.0
CCV
-1.4026168622179929
-0.301475
--4.652514
-0.000198
--2.035992
--0.769241
+0.275208
+-5.096563
+0.000075
+-1.980808
+-0.824425
CRV1
@@ -739,7 +739,7 @@ PyFixest 0.14.0
- Changes all docstrings to
numpy
format.
- Difference-in-differences estimation functions now need to be imported via the
pyfixest.did.estimation
module:
-
+
from pyfixest.did.estimation import did2s, lpdid, event_study
diff --git a/quarto_example/QuartoExample.pdf b/quarto_example/QuartoExample.pdf
index 78921f36..41692775 100644
Binary files a/quarto_example/QuartoExample.pdf and b/quarto_example/QuartoExample.pdf differ
diff --git a/quickstart.html b/quickstart.html
index 5bd8b2d2..7fca5546 100644
--- a/quickstart.html
+++ b/quickstart.html
@@ -281,7 +281,7 @@ What is a fix
Read Sample Data
In a first step, we load the module and some synthetic example data:
-
+
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
@@ -302,7 +302,7 @@ Read Sample Data
data.head()
-
+
@@ -336,7 +336,7 @@ Read Sample Data
-
+
@@ -370,7 +370,7 @@ Read Sample Data
-
+
-pandas : 2.2.3
-matplotlib: 3.9.2
-pyfixest : 0.24.3
+pyfixest : 0.24.3
numpy : 1.26.4
+matplotlib: 3.9.2
+pandas : 2.2.3
@@ -507,7 +507,7 @@ Read Sample Data
-
+
data.info()
<class 'pandas.core.frame.DataFrame'>
@@ -535,7 +535,7 @@ Read Sample Data
OLS Estimation
We are interested in the relation between the dependent variable Y
and the independent variables X1
using a fixed effect model for group_id
. Let’s see how the data looks like:
-
+
= data.plot(kind="scatter", x="X1", y="Y", c="group_id", colormap="viridis") ax
@@ -546,7 +546,7 @@ OLS Estimation
We can estimate a fixed effects regression via the feols()
function. feols()
has three arguments: a two-sided model formula, the data, and optionally, the type of inference.
-
+
= pf.feols(fml="Y ~ X1 | group_id", data=data, vcov="HC1")
fit type(fit)
@@ -559,7 +559,7 @@ OLS Estimation
Inspecting Model Results
To inspect the results, we can use a summary function or method:
-
+
fit.summary()
###
@@ -577,54 +577,54 @@ Inspecting Model
Or display a formatted regression table:
-
+
pf.etable(fit)
-
+
@@ -687,7 +687,7 @@ Inspecting Model
Alternatively, the .summarize
module contains a summary
function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable()
, please take a look at the dedicated vignette.
-
+
pf.summary(fit)
###
@@ -705,7 +705,7 @@ Inspecting Model
You can access individual elements of the summary via dedicated methods: .tidy()
returns a “tidy” pd.DataFrame
, .coef()
returns estimated parameters, and se()
estimated standard errors. Other methods include pvalue()
, confint()
and tstat()
.
-
+
fit.tidy()
@@ -748,7 +748,7 @@ Inspecting Model
-
+
fit.coef()
Coefficient
@@ -756,7 +756,7 @@ Inspecting Model
Name: Estimate, dtype: float64
-
+
fit.se()
Coefficient
@@ -764,7 +764,7 @@ Inspecting Model
Name: Std. Error, dtype: float64
-
+
fit.tstat()
Coefficient
@@ -772,7 +772,7 @@ Inspecting Model
Name: t value, dtype: float64
-
+
fit.confint()
@@ -799,11 +799,11 @@ Inspecting Model
Last, model results can be visualized via dedicated methods for plotting:
-
+
fit.coefplot()# or pf.coefplot([fit])
-
+
@@ -522,7 +522,7 @@ Examples
-
+
@@ -671,7 +671,7 @@ Examples
In a first step, we estimate a classical event study model:
-
+
# estimate the model
= pf.did2s(
fit
@@ -761,10 +761,10 @@ df_het,Examples
We can also inspect the model visually:
-
+
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -545,7 +545,7 @@ Examples
-
+
diff --git a/reference/did.estimation.lpdid.html b/reference/did.estimation.lpdid.html
index 6d7187ce..b74e15d2 100644
--- a/reference/did.estimation.lpdid.html
+++ b/reference/did.estimation.lpdid.html
@@ -505,7 +505,7 @@ Returns
Examples
-
+
import pandas as pd
import pyfixest as pf
@@ -528,7 +528,7 @@ Examples
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -562,7 +562,7 @@ Examples
-
+
-
+
@@ -588,7 +588,7 @@ Examples
-
+
@@ -638,7 +638,7 @@ Examples
Calling feols()
returns an instance of the [Feols(/reference/Feols.qmd) class. The summary()
method can be used to print the results.
An alternative way to retrieve model results is via the tidy()
method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.
-
+
fit.tidy()
@@ -692,17 +692,17 @@ Examples
You can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef()
for the coefficients, fit.se()
for the standard errors, fit.tstat()
for the t-statistics, and fit.pval()
for the p-values, and fit.confint()
for the confidence intervals.
The employed type of inference can be specified via the vcov
argument. If vcov is not provided, PyFixest
employs the fixest
default of iid inference, unless there are fixed effects in the model, in which case feols()
clusters the standard error by the first fixed effect (CRV1 inference).
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="iid")
fit1 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="hetero")
fit2 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1"}) fit3
Supported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {"CRV1": "f1"}
for CRV1 inference with clustering by f1
or {"CRV3": "f1"}
for CRV3 inference with clustering by f1
. For two-way clustering, you can provide a formula string, e.g. {"CRV1": "f1 + f2"}
for CRV1 inference with clustering by f1
.
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1 + f2"}) fit4
Inference can be adjusted post estimation via the vcov
method:
-
+
fit.summary()"iid").summary() fit.vcov(
@@ -736,7 +736,7 @@ Examples
The ssc
argument specifies the small sample correction for inference. In general, feols()
uses all of fixest::feols()
defaults, but sets the fixef.K
argument to "none"
whereas the fixest::feols()
default is "nested"
. See here for more details: link to github.
feols()
supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1
and one with fixed effects for f2
using the sw()
syntax.
-
+
= pf.feols("Y ~ X1 + X2 | sw(f1, f2)", data)
fit type(fit)
@@ -744,58 +744,58 @@ Examples
The returned object is an instance of the FixestMulti
class. You can access the results of the first model via fit.fetch_model(0)
and the results of the second model via fit.fetch_model(1)
. You can compare the model results via the etable()
function:
-
+
0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
Model: Y~X1+X2|f1
Model: Y~X1+X2|f2
-
+
@@ -837,14 +837,14 @@ Examples
fe
-f1
-x
+f2
-
+x
-f2
--
+f1
x
+-
stats
@@ -878,7 +878,7 @@ Examples
Other supported multiple estimation syntax include sw0()
, csw()
and csw0()
. While sw()
adds variables in a “stepwise” fashion, csw()
does so cumulatively.
-
+
= pf.feols("Y ~ X1 + X2 | csw(f1, f2)", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -886,51 +886,51 @@ Examples
Model: Y~X1+X2|f1+f2
-
+
@@ -972,13 +972,13 @@ Examples
fe
-f1
-x
+f2
+-
x
-f2
--
+f1
+x
x
@@ -1013,7 +1013,7 @@ Examples
The sw0()
and csw0()
syntax are similar to sw()
and csw()
, but start with a model that excludes the variables specified in sw()
and csw()
:
-
+
= pf.feols("Y ~ X1 + X2 | sw0(f1, f2)", data)
fit 0), fit.fetch_model(1), fit.fetch_model(2)]) pf.etable([fit.fetch_model(
@@ -1022,51 +1022,51 @@ Examples
Model: Y~X1+X2|f2
-
+
@@ -1121,16 +1121,16 @@ Examples
fe
-f1
+f2
-
-x
-
+x
-f2
--
+f1
-
x
+-
stats
@@ -1167,7 +1167,7 @@ Examples
The feols()
function also supports multiple dependent variables. The following example estimates two models, one with Y1
as the dependent variable and one with Y2
as the dependent variable.
-
+
= pf.feols("Y + Y2 ~ X1 | f1 + f2", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -1175,51 +1175,51 @@ Examples
Model: Y2~X1|f1+f2
-
+
@@ -1255,12 +1255,12 @@ Examples
fe
-f1
+f2
x
x
-f2
+f1
x
x
@@ -1296,7 +1296,7 @@ Examples
It is possible to combine different multiple estimation operators:
-
+
= pf.feols("Y + Y2 ~ X1 | sw(f1, f2)", data)
fit 0),
pf.etable([fit.fetch_model(1),
@@ -1311,51 +1311,51 @@ fit.fetch_model(Examples
Model: Y2~X1|f2
-
+
@@ -1401,18 +1401,18 @@ Examples
fe
-f1
-x
-x
+f2
-
-
+x
+x
-f2
--
--
+f1
x
x
+-
+-
stats
@@ -1453,7 +1453,7 @@ Examples
In general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols()
implements a caching mechanism that stores the demeaned covariates.
Besides OLS, feols()
also supports IV estimation via three part formulas:
-
+
= pf.feols("Y ~ X2 | f1 + f2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1507,7 +1507,7 @@ Examples
Here, X1
is the endogenous variable and Z1
is the instrument. f1
and f2
are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:
-
+
= pf.feols("Y ~ X2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1571,7 +1571,7 @@ Examples
Last, feols()
supports interaction of variables via the i()
syntax. Documentation on this is tba.
After fitting a model via feols()
, you can use the predict()
method to get the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit 0:5] fit.predict()[
@@ -1579,7 +1579,7 @@ Examples
The predict()
method also supports a newdata
argument to predict on new data, which returns a numpy array of the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit =data)[0:5] fit.predict(newdata
@@ -1587,11 +1587,11 @@ Examples
Last, you can plot the results of a model via the coefplot()
method:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit fit.coefplot()
-
+
@@ -569,7 +569,7 @@ Examples
-
+
diff --git a/reference/report.coefplot.html b/reference/report.coefplot.html
index d8796924..098ac83c 100644
--- a/reference/report.coefplot.html
+++ b/reference/report.coefplot.html
@@ -528,7 +528,7 @@ Returns
Examples
-
+
import pyfixest as pf
from pyfixest.report.utils import rename_categoricals
@@ -544,7 +544,7 @@ Examples
= "both") pf.coefplot([fit1], joint
-
+
@@ -578,7 +578,7 @@ Examples
-
+
-
+
@@ -576,7 +576,7 @@ Examples
-
+
-
+
@@ -497,7 +497,7 @@ Examples
-
+
diff --git a/replicating-the-effect.html b/replicating-the-effect.html
index 804007ca..a6a6a321 100644
--- a/replicating-the-effect.html
+++ b/replicating-the-effect.html
@@ -234,7 +234,7 @@ Replicating Examples from “The Effect”
This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.
-
+
from causaldata import Mroz, gapminder, organ_donations, restaurant_inspections
import pyfixest as pf
@@ -243,7 +243,7 @@ Replicating Examples from “The Effect”
%watermark --iversions
-
+
@@ -277,7 +277,7 @@ Replicating Examples from “The Effect”
-
+
-causaldata: 0.1.4
-pyfixest : 0.24.3
+pyfixest : 0.24.3
+causaldata: 0.1.4
Chapter 4: Describing Relationships
-
+
# Read in data
= Mroz.load_pandas().data
dt # Keep just working women
@@ -329,7 +329,7 @@ Chapter
= pf.feols(fml="lwg ~ csw(inc, wc, k5)", data=dt, vcov="iid")
fit pf.etable(fit)
-/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning:
+/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning:
A value is trying to be set on a copy of a slice from a DataFrame.
Try using .loc[row_indexer,col_indexer] = value instead
@@ -337,51 +337,51 @@ Chapter
dt.loc[:, "earn"] = dt["lwg"].apply("exp")
-
+
@@ -479,7 +479,7 @@ Chapter
Chapter 13: Regression
Example 1
-
+
= restaurant_inspections.load_pandas().data
res = res.inspection_score.astype(float)
res.inspection_score = res.NumberofLocations.astype(float)
@@ -488,51 +488,51 @@ res.NumberofLocations Example 1
= pf.feols(fml="inspection_score ~ NumberofLocations", data=res)
fit pf.etable([fit])
-
+
@@ -595,7 +595,7 @@ Example 1
Example 2
-
+
= restaurant_inspections.load_pandas().data
df
= pf.feols(
@@ -605,51 +605,51 @@ fit1 Example 2
pf.etable([fit1, fit2])
-
+
@@ -746,7 +746,7 @@ Example 2
Example 3: HC Standard Errors
-
+
="inspection_score ~ Year + Weekend", data=df, vcov="HC3").summary() pf.feols(fml
###
@@ -768,7 +768,7 @@ Example 3: HC
Example 4: Clustered Standard Errors
-
+
pf.feols(="inspection_score ~ Year + Weekend", data=df, vcov={"CRV1": "Weekend"}
fml ).tidy()
@@ -834,7 +834,7 @@ Exampl
Example 5: Bootstrap Inference
-
+
= pf.feols(fml="inspection_score ~ Year + Weekend", data=df)
fit =999, param="Year") fit.wildboottest(reps
@@ -857,7 +857,7 @@ Example 1
Example 2
-
+
= gapminder.load_pandas().data
gm "logGDPpercap"] = gm["gdpPercap"].apply("log")
gm[
@@ -943,7 +943,7 @@ Example 2
Example 3: TWFE
-
+
# Set our individual and time (index) for our data
= pf.feols(fml="lifeExp ~ np.log(gdpPercap) | country + year", data=gm)
fit fit.summary()
@@ -968,7 +968,7 @@ Example 3: TWFE
Chapter 18: Difference-in-Differences
Example 1
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
@@ -996,7 +996,7 @@ Example 1
Example 3: Dynamic Treatment Effect
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
diff --git a/search.json b/search.json
index f717fbdc..f69e7216 100644
--- a/search.json
+++ b/search.json
@@ -74,7 +74,7 @@
"href": "news.html#pyfixest-0.17.0",
"title": "News",
"section": "PyFixest 0.17.0",
- "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.379125\n1.178573\n\n\nD\n-1.759996\n-1.045238\n\n\nf1\n-0.014143\n0.023692\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.301475\n-4.652514\n0.000198\n-2.035992\n-0.769241\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
+ "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.373326\n1.184372\n\n\nD\n-1.765180\n-1.040053\n\n\nf1\n-0.014418\n0.023966\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.275208\n-5.096563\n0.000075\n-1.980808\n-0.824425\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
},
{
"objectID": "news.html#pyfixest-0.16.0",
@@ -298,7 +298,7 @@
"href": "quickstart.html",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#what-is-a-fixed-effect-model",
@@ -312,7 +312,7 @@
"href": "quickstart.html#read-sample-data",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
+ "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
},
{
"objectID": "quickstart.html#ols-estimation",
@@ -340,7 +340,7 @@
"href": "quickstart.html#updating-regression-coefficients",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#inference-via-the-wild-bootstrap",
@@ -375,7 +375,7 @@
"href": "quickstart.html#joint-confidence-intervals",
"title": "Getting Started with PyFixest",
"section": "Joint Confidence Intervals",
- "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.425936\n1.403847\n\n\nX1\n-1.160730\n-0.738152\n\n\nC(f1)[T.1.0]\n1.384234\n3.781064\n\n\nC(f1)[T.2.0]\n-2.838865\n-0.325003\n\n\nC(f1)[T.3.0]\n-1.608332\n0.983664"
+ "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.430485\n1.408396\n\n\nX1\n-1.161781\n-0.737102\n\n\nC(f1)[T.1.0]\n1.378276\n3.787022\n\n\nC(f1)[T.2.0]\n-2.845114\n-0.318754\n\n\nC(f1)[T.3.0]\n-1.614775\n0.990107"
},
{
"objectID": "contributing.html",
@@ -911,7 +911,7 @@
"href": "reference/estimation.estimation.feols.html#examples",
"title": "estimation.estimation.feols",
"section": "Examples",
- "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf1\nx\n-\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf1\n-\nx\n-\n\n\nf2\n-\n-\nx\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf1\nx\nx\n-\n-\n\n\nf2\n-\n-\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.265821\n0.060521\n0.952408\n-0.542381\n0.574556\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
+ "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\n-\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf2\n-\n-\nx\n\n\nf1\n-\nx\n-\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf2\nx\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf2\n-\n-\nx\nx\n\n\nf1\nx\nx\n-\n-\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.25692\n0.062617\n0.950761\n-0.523682\n0.555857\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
"crumbs": [
"Function Reference",
"Estimation Functions",
@@ -1435,14 +1435,14 @@
"href": "table-layout.html#basic-usage",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage",
- "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
+ "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
},
{
"objectID": "table-layout.html#keep-and-drop-variables",
"href": "table-layout.html#keep-and-drop-variables",
"title": "Regression Tables via pf.etable()",
"section": "Keep and drop variables",
- "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#hide-fixed-effects-or-se-type-rows",
@@ -1456,49 +1456,49 @@
"href": "table-layout.html#display-p-values-or-confidence-intervals",
"title": "Regression Tables via pf.etable()",
"section": "Display p-values or confidence intervals",
- "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#significance-levels-and-rounding",
"href": "table-layout.html#significance-levels-and-rounding",
"title": "Regression Tables via pf.etable()",
"section": "Significance levels and rounding",
- "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
+ "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
},
{
"objectID": "table-layout.html#other-output-formats",
"href": "table-layout.html#other-output-formats",
"title": "Regression Tables via pf.etable()",
"section": "Other output formats",
- "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf2 - x x - x x\nf1 x x x x x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
+ "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf1 x x x x x x\nf2 - x x - x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
},
{
"objectID": "table-layout.html#rename-variables",
"href": "table-layout.html#rename-variables",
"title": "Regression Tables via pf.etable()",
"section": "Rename variables",
- "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-model-headlines",
"href": "table-layout.html#custom-model-headlines",
"title": "Regression Tables via pf.etable()",
"section": "Custom model headlines",
- "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
+ "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
},
{
"objectID": "table-layout.html#further-custom-model-information",
"href": "table-layout.html#further-custom-model-information",
"title": "Regression Tables via pf.etable()",
"section": "Further custom model information",
- "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-table-notes",
"href": "table-layout.html#custom-table-notes",
"title": "Regression Tables via pf.etable()",
"section": "Custom table notes",
- "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
+ "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
},
{
"objectID": "table-layout.html#publication-ready-latex-tables",
@@ -1519,49 +1519,49 @@
"href": "table-layout.html#summarize-by-characteristics-in-columns-and-rows",
"title": "Regression Tables via pf.etable()",
"section": "Summarize by characteristics in columns and rows",
- "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.19\n 2.35\n -0.04\n 2.21\n -0.05\n 2.28\n -0.23\n 2.39\n \n \n Wealth\n -0.72\n 5.71\n 0.05\n 4.99\n -0.09\n 5.53\n -0.56\n 6.06\n \n \n Age\n 1.07\n 0.81\n 0.96\n 0.80\n 1.08\n 0.81\n 1.06\n 0.81\n \n \n Years of Schooling\n 0.05\n 3.10\n -0.28\n 2.86\n -0.18\n 3.13\n -0.09\n 3.08\n \n \n nobs\n \n \n Number of observations\n 229\n \n 244\n \n 270\n \n 254\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.19(2.35)\n -0.04(2.21)\n -0.05(2.28)\n -0.23(2.39)\n \n \n Wealth\n -0.72(5.71)\n 0.05(4.99)\n -0.09(5.53)\n -0.56(6.06)\n \n \n Age\n 1.07(0.81)\n 0.96(0.80)\n 1.08(0.81)\n 1.06(0.81)\n \n \n Years of Schooling\n 0.05(3.10)\n -0.28(2.86)\n -0.18(3.13)\n -0.09(3.08)\n \n \n nobs\n \n \n Number of observations\n 229\n 244\n 270\n 254\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 229\n -0.19\n 2.35\n 270\n -0.05\n 2.28\n \n \n Wealth\n 229\n -0.72\n 5.71\n 270\n -0.09\n 5.53\n \n \n Age\n 229\n 1.07\n 0.81\n 270\n 1.08\n 0.81\n \n \n Years of Schooling\n 229\n 0.05\n 3.10\n 270\n -0.18\n 3.13\n \n \n White collar\n \n \n Wage\n 244\n -0.04\n 2.21\n 254\n -0.23\n 2.39\n \n \n Wealth\n 244\n 0.05\n 4.99\n 254\n -0.56\n 6.06\n \n \n Age\n 244\n 0.96\n 0.80\n 254\n 1.06\n 0.81\n \n \n Years of Schooling\n 244\n -0.28\n 2.86\n 254\n -0.09\n 3.08\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
+ "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.18\n 2.35\n 0.10\n 2.26\n -0.28\n 2.33\n -0.13\n 2.28\n \n \n Wealth\n -0.33\n 5.51\n -0.10\n 5.54\n -0.39\n 5.48\n -0.45\n 5.83\n \n \n Age\n 1.05\n 0.82\n 1.05\n 0.81\n 1.02\n 0.81\n 1.05\n 0.80\n \n \n Years of Schooling\n -0.14\n 2.99\n -0.31\n 2.92\n 0.12\n 3.24\n -0.19\n 3.02\n \n \n nobs\n \n \n Number of observations\n 254\n \n 242\n \n 246\n \n 255\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.18(2.35)\n 0.10(2.26)\n -0.28(2.33)\n -0.13(2.28)\n \n \n Wealth\n -0.33(5.51)\n -0.10(5.54)\n -0.39(5.48)\n -0.45(5.83)\n \n \n Age\n 1.05(0.82)\n 1.05(0.81)\n 1.02(0.81)\n 1.05(0.80)\n \n \n Years of Schooling\n -0.14(2.99)\n -0.31(2.92)\n 0.12(3.24)\n -0.19(3.02)\n \n \n nobs\n \n \n Number of observations\n 254\n 242\n 246\n 255\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 254\n -0.18\n 2.35\n 246\n -0.28\n 2.33\n \n \n Wealth\n 254\n -0.33\n 5.51\n 246\n -0.39\n 5.48\n \n \n Age\n 254\n 1.05\n 0.82\n 246\n 1.02\n 0.81\n \n \n Years of Schooling\n 254\n -0.14\n 2.99\n 246\n 0.12\n 3.24\n \n \n White collar\n \n \n Wage\n 242\n 0.10\n 2.26\n 255\n -0.13\n 2.28\n \n \n Wealth\n 242\n -0.10\n 5.54\n 255\n -0.45\n 5.83\n \n \n Age\n 242\n 1.05\n 0.81\n 255\n 1.05\n 0.80\n \n \n Years of Schooling\n 242\n -0.31\n 2.92\n 255\n -0.19\n 3.02\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
},
{
"objectID": "table-layout.html#basic-usage-of-make_table",
"href": "table-layout.html#basic-usage-of-make_table",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage of make_table",
- "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n -0.04\n -1.77\n -1.68\n 1.89\n \n \n 1\n 0.39\n 1.05\n -0.07\n -1.59\n \n \n 2\n 1.1\n -0.18\n 0.3\n -1.25\n \n \n 3\n 1.78\n -0.18\n -0.09\n -0.08\n \n\n \n \n \n These are notes"
+ "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n 0.87\n -1.25\n -1.78\n 1.12\n \n \n 1\n -0.88\n 1.08\n -0.47\n -0.5\n \n \n 2\n -0.31\n 1.04\n 0.56\n 0.12\n \n \n 3\n 0.37\n -0.63\n -0.72\n 1.4\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#mutiindex-dataframes",
"href": "table-layout.html#mutiindex-dataframes",
"title": "Regression Tables via pf.etable()",
"section": "Mutiindex DataFrames",
- "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes"
+ "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#example-styling",
"href": "table-layout.html#example-styling",
"title": "Regression Tables via pf.etable()",
"section": "Example Styling",
- "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#defining-table-styles-some-examples",
"href": "table-layout.html#defining-table-styles-some-examples",
"title": "Regression Tables via pf.etable()",
"section": "Defining Table Styles: Some Examples",
- "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "replicating-the-effect.html",
"href": "replicating-the-effect.html",
"title": "Replicating Examples from “The Effect”",
"section": "",
- "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\ncausaldata: 0.1.4\npyfixest : 0.24.3"
+ "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\ncausaldata: 0.1.4"
},
{
"objectID": "replicating-the-effect.html#chapter-4-describing-relationships",
"href": "replicating-the-effect.html#chapter-4-describing-relationships",
"title": "Replicating Examples from “The Effect”",
"section": "Chapter 4: Describing Relationships",
- "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
+ "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
},
{
"objectID": "replicating-the-effect.html#chapter-13-regression",
@@ -1624,7 +1624,7 @@
"href": "difference-in-differences.html#inspecting-the-outcome-variable",
"title": "Difference-in-Differences Estimation",
"section": "Inspecting the Outcome Variable",
- "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)"
+ "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n title = \"Outcome Plot\"\n)"
},
{
"objectID": "difference-in-differences.html#one-shot-adoption-static-and-dynamic-specifications",
diff --git a/table-layout.html b/table-layout.html
index 3dd0c424..2773792b 100644
--- a/table-layout.html
+++ b/table-layout.html
@@ -245,7 +245,7 @@ Regression Tables via pf.etable()
Table Layout with PyFixest
Pyfixest comes with functions to generate publication-ready tables. Regression tables are generated with pf.etable()
, which can output different formats, for instance using the Great Tables package or generating formatted LaTex Tables using booktabs. There are also further functions pf.dtable()
to display descriptive statistics and pf.make_table()
generating formatted tables from pandas dataframes in the same layout.
To begin, we load some libraries and fit a set of regression models.
-
+
import numpy as np
import pandas as pd
import pylatex as pl # for the latex table; note: not a dependency of pyfixest - needs manual installation
@@ -267,7 +267,7 @@ Table Layout wi
= pf.feols("Y2 ~ X1 *X2 | f1 + f2", data=data) fit6
-
+
@@ -301,7 +301,7 @@ Table Layout wi
-
+
@@ -338,54 +338,54 @@ Table Layout wi
Basic Usage
We can compare all regression models via the pyfixest-internal pf.etable()
function:
-
+
pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
-
+
@@ -444,20 +444,20 @@ Basic Usage
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -507,54 +507,54 @@ Basic Usage
You can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:
-
+
"Y+Y2~csw(X1,X2,X1:X2)", data=data)) pf.etable(pf.feols(
-
+
@@ -667,54 +667,54 @@ Basic Usage
Keep and drop variables
etable
allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.
-
+
="X1") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -764,20 +764,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -827,54 +827,54 @@ Keep and drop vari
We can use the exact_match
argument to select a specific set of variables:
-
+
=["X1", "X2"], exact_match=True) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -924,20 +924,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -987,54 +987,54 @@ Keep and drop vari
We can also easily drop variables via the drop
argument:
-
+
=["X1"]) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop
-
+
@@ -1075,20 +1075,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1141,54 +1141,54 @@ Keep and drop vari
Hide fixed effects or SE-type rows
We can hide the rows showing the relevant fixed effects and those showing the S.E. type by setting show_fe=False
and show_setype=False
(for instance when the set of fixed effects or the estimation method for the std. errors is the same for all models and you want to describe this in the text or table notes rather than displaying it in the table).
-
+
=False, show_se_type=False) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], show_fe
-
+
@@ -1283,54 +1283,54 @@ Hide fi
Display p-values or confidence intervals
By default, pf.etable()
reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt
function argument.
-
+
="b \n (se) \n [p]") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt
-
+
@@ -1389,20 +1389,20 @@ D
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1455,54 +1455,54 @@ D
Significance levels and rounding
Additionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code
and digits
function arguments:
-
+
=[0.01, 0.05, 0.1], digits=5) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code
-
+
@@ -1561,20 +1561,20 @@ Significa
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1627,7 +1627,7 @@ Significa
Other output formats
By default, pf.etable()
returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type
argument.
-
+
# Pandas styler output:
pf.etable(
@@ -1689,20 +1689,20 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
-0.04082 (0.08093)
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1738,7 +1738,7 @@ Other output formats<
-
+
# Markdown output:
pf.etable(
@@ -1758,8 +1758,8 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
X1:X2 0.01057 -0.04082
(0.01818) (0.08093)
------------------------------------------------------------------------------------------------
-f2 - x x - x x
f1 x x x x x x
+f2 - x x - x x
------------------------------------------------------------------------------------------------
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
@@ -1769,7 +1769,7 @@ Other output formats<
To obtain latex output use format = "tex"
. If you want to save the table as a tex file, you can use the filename=
argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True
argument. Etable will use latex packages booktabs
, threeparttable
and makecell
for the table layout, so don’t forget to include these packages in your latex document.
-
+
# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):
= pf.etable(
tab
@@ -1780,7 +1780,7 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
)
The following code generates a pdf including the regression table which you can display clicking on the link below the cell:
-
+
## Use pylatex to create a tex file with the table
@@ -1814,7 +1814,7 @@ Other output formats<
Rename variables
You can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels
argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).
-
+
Cluster-Robust Errors
We conclude with cluster robust errors.
-= pf.feols(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"})
fit = pf.feols(
fit_weights ="Y ~ X1 + X2 | f1 + f2", data=data, vcov={"CRV1": "f1"}, weights="weights"
@@ -821,14 +821,14 @@ fmlCluster-Robust Error
- stats.vcov(r_fit) fit._vcov
array([[ 4.20670443e-16, -6.97565513e-17],
[-6.97565513e-17, -1.42166010e-17]])
- stats.vcov(r_fit_weights) fit_weights._vcov
array([[2.59070109e-16, 4.07324592e-16],
@@ -836,7 +836,7 @@ Cluster-Robust Error
We conclude by comparing all estimation results via the tidy
methods:
fit.tidy()
Cluster-Robust Error
pd.DataFrame(broom.tidy_fixest(r_fit)).T
Cluster-Robust Error
fit_weights.tidy()
Cluster-Robust Error
pd.DataFrame(broom.tidy_fixest(r_fit_weights)).T
Cluster-Robust Error
Poisson Regression
-
+
= pf.get_data(model="Fepois") data
-
+
= pf.fepois(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid", iwls_tol=1e-10)
fit_iid = pf.fepois(
fit_hetero ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", iwls_tol=1e-10
@@ -1065,21 +1065,21 @@ fmlPoisson Regression
-
+
- stats.vcov(fit_r_iid) fit_iid._vcov
array([[ 1.20791284e-08, -6.55604931e-10],
[-6.55604931e-10, 1.69958097e-09]])
-
+
- stats.vcov(fit_r_hetero) fit_hetero._vcov
array([[ 2.18101847e-08, -7.38711972e-10],
[-7.38711972e-10, 3.07587753e-09]])
-
+
- stats.vcov(fit_r_crv) fit_crv._vcov
array([[ 1.58300904e-08, -1.20806815e-10],
@@ -1087,7 +1087,7 @@ Poisson Regression
We conclude by comparing all estimation results via the tidy
methods:
-
+
fit_iid.tidy()
@@ -1139,7 +1139,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_iid)).T
@@ -1179,7 +1179,7 @@ Poisson Regression
-
+
fit_hetero.tidy()
@@ -1231,7 +1231,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_hetero)).T
@@ -1271,7 +1271,7 @@ Poisson Regression
-
+
fit_crv.tidy()
@@ -1323,7 +1323,7 @@ Poisson Regression
-
+
pd.DataFrame(broom.tidy_fixest(fit_r_crv)).T
diff --git a/difference-in-differences.html b/difference-in-differences.html
index 9fdf3693..e42777c0 100644
--- a/difference-in-differences.html
+++ b/difference-in-differences.html
@@ -257,7 +257,7 @@ Difference-in-Differences Estimation
See also NBER SI methods lectures on Linear Panel Event Studies.
Setup
-
+
from importlib import resources
import pandas as pd
@@ -272,7 +272,7 @@ Setup
%autoreload 2
-
+
@@ -306,7 +306,7 @@ Setup
-
+
@@ -344,7 +344,7 @@ Setup
-
+
# one-shot adoption data - parallel trends is true
= get_sharkfin()
df_one_cohort df_one_cohort.head()
@@ -410,7 +410,7 @@ Setup
-
+
# multi-cohort adoption data
= pd.read_csv(
df_multi_cohort "pyfixest.did.data").joinpath("df_het.csv")
@@ -536,7 +536,7 @@ resources.files(Setup
Examining Treatment Timing
Before any DiD estimation, we need to examine the treatment timing, since it is crucial to our choice of estimator.
-
+
pf.panelview(
df_one_cohort,="unit",
@@ -557,7 +557,7 @@ unitExamining Treat
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -580,7 +580,7 @@ unitExamining Treat
We immediately see that we have staggered adoption of treatment in the second case, which implies that a naive application of 2WFE might yield biased estimates under substantial effect heterogeneity.
We can also plot treatment assignment in a disaggregated fashion, which gives us a sense of cohort sizes.
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -604,7 +604,7 @@ unitExamining Treat
Inspecting the Outcome Variable
pf.panelview()
further allows us to inspect the “outcome” variable over time:
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -612,20 +612,19 @@ outcomeInspecting
="year",
time="treat",
treat=True,
- collapse_to_cohort=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
We immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.
We can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -633,13 +632,12 @@ outcomeInspecting
="year",
time="treat",
treat=100,
- subsamp=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
@@ -648,7 +646,7 @@ Inspecting
One-shot adoption: Static and Dynamic Specifications
After taking a first look at the data, let’s turn to estimation. We return to the df_one_cohort
data set (without staggered treatment rollout).
-
+
= pf.feols(
fit_static_twfe "Y ~ treat | unit + year",
@@ -671,14 +669,14 @@ df_one_cohort,
+
= pf.feols(
fit_dynamic_twfe "Y ~ i(year, ever_treated, ref = 14) | unit + year",
df_one_cohort,={"CRV1": "unit"},
vcov )
-
+
fit_dynamic_twfe.iplot(=False,
coord_flip="Event Study",
@@ -688,7 +686,7 @@ title=rename_event_study_coefs(fit_dynamic_twfe._coefnames),
)
labels
-
+
-
+
fit_lpdid.iplot(=False,
coord_flip="Local-Projections-Estimator",
@@ -1167,7 +1165,7 @@ titleLocal Project
=18.5,
xintercept ).show()
-
+
@@ -297,7 +297,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
-
+
@@ -390,7 +390,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
Suppose we were interested in testing the hypothesis that \(X_{1} = X_{2}\). Given the relatively large differences in coefficients and small standard errors, we will likely reject the null that the two parameters are equal.
We can run the formal test via the hypotheses
function from the marginaleffects
package.
-
+
"X1 - X2 = 0") hypotheses(fit,
@@ -545,7 +545,7 @@ PyFixest 0.18.0
Additionally, model_matrix_fixest
now returns a dictionary instead of a tuple.
Brings back fixed effects reference setting via i(var1, var2, ref)
syntax. Deprecates the i_ref1
, i_ref2
function arguments. I.e. it is again possible to e.g. run
-
+
import pyfixest as pf
= pf.get_data()
data
@@ -553,7 +553,7 @@ PyFixest 0.18.0
0:8] fit1.coef()[
Via the ref
syntax, via can set the reference level:
-
+
= pf.feols("Y ~ i(f1, X2, ref = 1)", data=data)
fit2 0:8] fit2.coef()[
@@ -562,7 +562,7 @@ PyFixest 0.18.0
PyFixest 0.17.0
Restructures the codebase and reorganizes how users can interact with the pyfixest
API. It is now recommended to use pyfixest
in the following way:
-
+
import numpy as np
import pyfixest as pf
= pf.get_data()
@@ -630,7 +630,7 @@ data PyFixest 0.17.0
The update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!
Adds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!
-
+
= True) fit.confint(joint
@@ -647,18 +647,18 @@ PyFixest 0.17.0
Intercept
-0.379125
-1.178573
+0.373326
+1.184372
D
--1.759996
--1.045238
+-1.765180
+-1.040053
f1
--0.014143
-0.023692
+-0.014418
+0.023966
@@ -667,7 +667,7 @@ PyFixest 0.17.0
Adds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv()
method.
-
+
= "D", cluster = "group_id") fit.ccv(treatment
/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.
@@ -693,11 +693,11 @@ PyFixest 0.17.0
CCV
-1.4026168622179929
-0.301475
--4.652514
-0.000198
--2.035992
--0.769241
+0.275208
+-5.096563
+0.000075
+-1.980808
+-0.824425
CRV1
@@ -739,7 +739,7 @@ PyFixest 0.14.0
- Changes all docstrings to
numpy
format.
- Difference-in-differences estimation functions now need to be imported via the
pyfixest.did.estimation
module:
-
+
from pyfixest.did.estimation import did2s, lpdid, event_study
diff --git a/quarto_example/QuartoExample.pdf b/quarto_example/QuartoExample.pdf
index 78921f36..41692775 100644
Binary files a/quarto_example/QuartoExample.pdf and b/quarto_example/QuartoExample.pdf differ
diff --git a/quickstart.html b/quickstart.html
index 5bd8b2d2..7fca5546 100644
--- a/quickstart.html
+++ b/quickstart.html
@@ -281,7 +281,7 @@ What is a fix
Read Sample Data
In a first step, we load the module and some synthetic example data:
-
+
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
@@ -302,7 +302,7 @@ Read Sample Data
data.head()
-
+
@@ -336,7 +336,7 @@ Read Sample Data
-
+
@@ -370,7 +370,7 @@ Read Sample Data
-
+
-pandas : 2.2.3
-matplotlib: 3.9.2
-pyfixest : 0.24.3
+pyfixest : 0.24.3
numpy : 1.26.4
+matplotlib: 3.9.2
+pandas : 2.2.3
@@ -507,7 +507,7 @@ Read Sample Data
-
+
data.info()
<class 'pandas.core.frame.DataFrame'>
@@ -535,7 +535,7 @@ Read Sample Data
OLS Estimation
We are interested in the relation between the dependent variable Y
and the independent variables X1
using a fixed effect model for group_id
. Let’s see how the data looks like:
-
+
= data.plot(kind="scatter", x="X1", y="Y", c="group_id", colormap="viridis") ax
@@ -546,7 +546,7 @@ OLS Estimation
We can estimate a fixed effects regression via the feols()
function. feols()
has three arguments: a two-sided model formula, the data, and optionally, the type of inference.
-
+
= pf.feols(fml="Y ~ X1 | group_id", data=data, vcov="HC1")
fit type(fit)
@@ -559,7 +559,7 @@ OLS Estimation
Inspecting Model Results
To inspect the results, we can use a summary function or method:
-
+
fit.summary()
###
@@ -577,54 +577,54 @@ Inspecting Model
Or display a formatted regression table:
-
+
pf.etable(fit)
-
+
@@ -687,7 +687,7 @@ Inspecting Model
Alternatively, the .summarize
module contains a summary
function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable()
, please take a look at the dedicated vignette.
-
+
pf.summary(fit)
###
@@ -705,7 +705,7 @@ Inspecting Model
You can access individual elements of the summary via dedicated methods: .tidy()
returns a “tidy” pd.DataFrame
, .coef()
returns estimated parameters, and se()
estimated standard errors. Other methods include pvalue()
, confint()
and tstat()
.
-
+
fit.tidy()
@@ -748,7 +748,7 @@ Inspecting Model
-
+
fit.coef()
Coefficient
@@ -756,7 +756,7 @@ Inspecting Model
Name: Estimate, dtype: float64
-
+
fit.se()
Coefficient
@@ -764,7 +764,7 @@ Inspecting Model
Name: Std. Error, dtype: float64
-
+
fit.tstat()
Coefficient
@@ -772,7 +772,7 @@ Inspecting Model
Name: t value, dtype: float64
-
+
fit.confint()
@@ -799,11 +799,11 @@ Inspecting Model
Last, model results can be visualized via dedicated methods for plotting:
-
+
fit.coefplot()# or pf.coefplot([fit])
-
+
@@ -522,7 +522,7 @@ Examples
-
+
@@ -671,7 +671,7 @@ Examples
In a first step, we estimate a classical event study model:
-
+
# estimate the model
= pf.did2s(
fit
@@ -761,10 +761,10 @@ df_het,Examples
We can also inspect the model visually:
-
+
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -545,7 +545,7 @@ Examples
-
+
diff --git a/reference/did.estimation.lpdid.html b/reference/did.estimation.lpdid.html
index 6d7187ce..b74e15d2 100644
--- a/reference/did.estimation.lpdid.html
+++ b/reference/did.estimation.lpdid.html
@@ -505,7 +505,7 @@ Returns
Examples
-
+
import pandas as pd
import pyfixest as pf
@@ -528,7 +528,7 @@ Examples
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -562,7 +562,7 @@ Examples
-
+
-
+
@@ -588,7 +588,7 @@ Examples
-
+
@@ -638,7 +638,7 @@ Examples
Calling feols()
returns an instance of the [Feols(/reference/Feols.qmd) class. The summary()
method can be used to print the results.
An alternative way to retrieve model results is via the tidy()
method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.
-
+
fit.tidy()
@@ -692,17 +692,17 @@ Examples
You can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef()
for the coefficients, fit.se()
for the standard errors, fit.tstat()
for the t-statistics, and fit.pval()
for the p-values, and fit.confint()
for the confidence intervals.
The employed type of inference can be specified via the vcov
argument. If vcov is not provided, PyFixest
employs the fixest
default of iid inference, unless there are fixed effects in the model, in which case feols()
clusters the standard error by the first fixed effect (CRV1 inference).
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="iid")
fit1 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="hetero")
fit2 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1"}) fit3
Supported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {"CRV1": "f1"}
for CRV1 inference with clustering by f1
or {"CRV3": "f1"}
for CRV3 inference with clustering by f1
. For two-way clustering, you can provide a formula string, e.g. {"CRV1": "f1 + f2"}
for CRV1 inference with clustering by f1
.
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1 + f2"}) fit4
Inference can be adjusted post estimation via the vcov
method:
-
+
fit.summary()"iid").summary() fit.vcov(
@@ -736,7 +736,7 @@ Examples
The ssc
argument specifies the small sample correction for inference. In general, feols()
uses all of fixest::feols()
defaults, but sets the fixef.K
argument to "none"
whereas the fixest::feols()
default is "nested"
. See here for more details: link to github.
feols()
supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1
and one with fixed effects for f2
using the sw()
syntax.
-
+
= pf.feols("Y ~ X1 + X2 | sw(f1, f2)", data)
fit type(fit)
@@ -744,58 +744,58 @@ Examples
The returned object is an instance of the FixestMulti
class. You can access the results of the first model via fit.fetch_model(0)
and the results of the second model via fit.fetch_model(1)
. You can compare the model results via the etable()
function:
-
+
0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
Model: Y~X1+X2|f1
Model: Y~X1+X2|f2
-
+
@@ -837,14 +837,14 @@ Examples
fe
-f1
-x
+f2
-
+x
-f2
--
+f1
x
+-
stats
@@ -878,7 +878,7 @@ Examples
Other supported multiple estimation syntax include sw0()
, csw()
and csw0()
. While sw()
adds variables in a “stepwise” fashion, csw()
does so cumulatively.
-
+
= pf.feols("Y ~ X1 + X2 | csw(f1, f2)", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -886,51 +886,51 @@ Examples
Model: Y~X1+X2|f1+f2
-
+
@@ -972,13 +972,13 @@ Examples
fe
-f1
-x
+f2
+-
x
-f2
--
+f1
+x
x
@@ -1013,7 +1013,7 @@ Examples
The sw0()
and csw0()
syntax are similar to sw()
and csw()
, but start with a model that excludes the variables specified in sw()
and csw()
:
-
+
= pf.feols("Y ~ X1 + X2 | sw0(f1, f2)", data)
fit 0), fit.fetch_model(1), fit.fetch_model(2)]) pf.etable([fit.fetch_model(
@@ -1022,51 +1022,51 @@ Examples
Model: Y~X1+X2|f2
-
+
@@ -1121,16 +1121,16 @@ Examples
fe
-f1
+f2
-
-x
-
+x
-f2
--
+f1
-
x
+-
stats
@@ -1167,7 +1167,7 @@ Examples
The feols()
function also supports multiple dependent variables. The following example estimates two models, one with Y1
as the dependent variable and one with Y2
as the dependent variable.
-
+
= pf.feols("Y + Y2 ~ X1 | f1 + f2", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -1175,51 +1175,51 @@ Examples
Model: Y2~X1|f1+f2
-
+
@@ -1255,12 +1255,12 @@ Examples
fe
-f1
+f2
x
x
-f2
+f1
x
x
@@ -1296,7 +1296,7 @@ Examples
It is possible to combine different multiple estimation operators:
-
+
= pf.feols("Y + Y2 ~ X1 | sw(f1, f2)", data)
fit 0),
pf.etable([fit.fetch_model(1),
@@ -1311,51 +1311,51 @@ fit.fetch_model(Examples
Model: Y2~X1|f2
-
+
@@ -1401,18 +1401,18 @@ Examples
fe
-f1
-x
-x
+f2
-
-
+x
+x
-f2
--
--
+f1
x
x
+-
+-
stats
@@ -1453,7 +1453,7 @@ Examples
In general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols()
implements a caching mechanism that stores the demeaned covariates.
Besides OLS, feols()
also supports IV estimation via three part formulas:
-
+
= pf.feols("Y ~ X2 | f1 + f2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1507,7 +1507,7 @@ Examples
Here, X1
is the endogenous variable and Z1
is the instrument. f1
and f2
are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:
-
+
= pf.feols("Y ~ X2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1571,7 +1571,7 @@ Examples
Last, feols()
supports interaction of variables via the i()
syntax. Documentation on this is tba.
After fitting a model via feols()
, you can use the predict()
method to get the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit 0:5] fit.predict()[
@@ -1579,7 +1579,7 @@ Examples
The predict()
method also supports a newdata
argument to predict on new data, which returns a numpy array of the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit =data)[0:5] fit.predict(newdata
@@ -1587,11 +1587,11 @@ Examples
Last, you can plot the results of a model via the coefplot()
method:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit fit.coefplot()
-
+
@@ -569,7 +569,7 @@ Examples
-
+
diff --git a/reference/report.coefplot.html b/reference/report.coefplot.html
index d8796924..098ac83c 100644
--- a/reference/report.coefplot.html
+++ b/reference/report.coefplot.html
@@ -528,7 +528,7 @@ Returns
Examples
-
+
import pyfixest as pf
from pyfixest.report.utils import rename_categoricals
@@ -544,7 +544,7 @@ Examples
= "both") pf.coefplot([fit1], joint
-
+
@@ -578,7 +578,7 @@ Examples
-
+
-
+
@@ -576,7 +576,7 @@ Examples
-
+
-
+
@@ -497,7 +497,7 @@ Examples
-
+
diff --git a/replicating-the-effect.html b/replicating-the-effect.html
index 804007ca..a6a6a321 100644
--- a/replicating-the-effect.html
+++ b/replicating-the-effect.html
@@ -234,7 +234,7 @@ Replicating Examples from “The Effect”
This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.
-
+
from causaldata import Mroz, gapminder, organ_donations, restaurant_inspections
import pyfixest as pf
@@ -243,7 +243,7 @@ Replicating Examples from “The Effect”
%watermark --iversions
-
+
@@ -277,7 +277,7 @@ Replicating Examples from “The Effect”
-
+
-causaldata: 0.1.4
-pyfixest : 0.24.3
+pyfixest : 0.24.3
+causaldata: 0.1.4
Chapter 4: Describing Relationships
-
+
# Read in data
= Mroz.load_pandas().data
dt # Keep just working women
@@ -329,7 +329,7 @@ Chapter
= pf.feols(fml="lwg ~ csw(inc, wc, k5)", data=dt, vcov="iid")
fit pf.etable(fit)
-/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning:
+/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning:
A value is trying to be set on a copy of a slice from a DataFrame.
Try using .loc[row_indexer,col_indexer] = value instead
@@ -337,51 +337,51 @@ Chapter
dt.loc[:, "earn"] = dt["lwg"].apply("exp")
-
+
@@ -479,7 +479,7 @@ Chapter
Chapter 13: Regression
Example 1
-
+
= restaurant_inspections.load_pandas().data
res = res.inspection_score.astype(float)
res.inspection_score = res.NumberofLocations.astype(float)
@@ -488,51 +488,51 @@ res.NumberofLocations Example 1
= pf.feols(fml="inspection_score ~ NumberofLocations", data=res)
fit pf.etable([fit])
-
+
@@ -595,7 +595,7 @@ Example 1
Example 2
-
+
= restaurant_inspections.load_pandas().data
df
= pf.feols(
@@ -605,51 +605,51 @@ fit1 Example 2
pf.etable([fit1, fit2])
-
+
@@ -746,7 +746,7 @@ Example 2
Example 3: HC Standard Errors
-
+
="inspection_score ~ Year + Weekend", data=df, vcov="HC3").summary() pf.feols(fml
###
@@ -768,7 +768,7 @@ Example 3: HC
Example 4: Clustered Standard Errors
-
+
pf.feols(="inspection_score ~ Year + Weekend", data=df, vcov={"CRV1": "Weekend"}
fml ).tidy()
@@ -834,7 +834,7 @@ Exampl
Example 5: Bootstrap Inference
-
+
= pf.feols(fml="inspection_score ~ Year + Weekend", data=df)
fit =999, param="Year") fit.wildboottest(reps
@@ -857,7 +857,7 @@ Example 1
Example 2
-
+
= gapminder.load_pandas().data
gm "logGDPpercap"] = gm["gdpPercap"].apply("log")
gm[
@@ -943,7 +943,7 @@ Example 2
Example 3: TWFE
-
+
# Set our individual and time (index) for our data
= pf.feols(fml="lifeExp ~ np.log(gdpPercap) | country + year", data=gm)
fit fit.summary()
@@ -968,7 +968,7 @@ Example 3: TWFE
Chapter 18: Difference-in-Differences
Example 1
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
@@ -996,7 +996,7 @@ Example 1
Example 3: Dynamic Treatment Effect
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
diff --git a/search.json b/search.json
index f717fbdc..f69e7216 100644
--- a/search.json
+++ b/search.json
@@ -74,7 +74,7 @@
"href": "news.html#pyfixest-0.17.0",
"title": "News",
"section": "PyFixest 0.17.0",
- "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.379125\n1.178573\n\n\nD\n-1.759996\n-1.045238\n\n\nf1\n-0.014143\n0.023692\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.301475\n-4.652514\n0.000198\n-2.035992\n-0.769241\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
+ "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.373326\n1.184372\n\n\nD\n-1.765180\n-1.040053\n\n\nf1\n-0.014418\n0.023966\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.275208\n-5.096563\n0.000075\n-1.980808\n-0.824425\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
},
{
"objectID": "news.html#pyfixest-0.16.0",
@@ -298,7 +298,7 @@
"href": "quickstart.html",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#what-is-a-fixed-effect-model",
@@ -312,7 +312,7 @@
"href": "quickstart.html#read-sample-data",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
+ "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
},
{
"objectID": "quickstart.html#ols-estimation",
@@ -340,7 +340,7 @@
"href": "quickstart.html#updating-regression-coefficients",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#inference-via-the-wild-bootstrap",
@@ -375,7 +375,7 @@
"href": "quickstart.html#joint-confidence-intervals",
"title": "Getting Started with PyFixest",
"section": "Joint Confidence Intervals",
- "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.425936\n1.403847\n\n\nX1\n-1.160730\n-0.738152\n\n\nC(f1)[T.1.0]\n1.384234\n3.781064\n\n\nC(f1)[T.2.0]\n-2.838865\n-0.325003\n\n\nC(f1)[T.3.0]\n-1.608332\n0.983664"
+ "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.430485\n1.408396\n\n\nX1\n-1.161781\n-0.737102\n\n\nC(f1)[T.1.0]\n1.378276\n3.787022\n\n\nC(f1)[T.2.0]\n-2.845114\n-0.318754\n\n\nC(f1)[T.3.0]\n-1.614775\n0.990107"
},
{
"objectID": "contributing.html",
@@ -911,7 +911,7 @@
"href": "reference/estimation.estimation.feols.html#examples",
"title": "estimation.estimation.feols",
"section": "Examples",
- "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf1\nx\n-\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf1\n-\nx\n-\n\n\nf2\n-\n-\nx\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf1\nx\nx\n-\n-\n\n\nf2\n-\n-\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.265821\n0.060521\n0.952408\n-0.542381\n0.574556\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
+ "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\n-\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf2\n-\n-\nx\n\n\nf1\n-\nx\n-\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf2\nx\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf2\n-\n-\nx\nx\n\n\nf1\nx\nx\n-\n-\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.25692\n0.062617\n0.950761\n-0.523682\n0.555857\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
"crumbs": [
"Function Reference",
"Estimation Functions",
@@ -1435,14 +1435,14 @@
"href": "table-layout.html#basic-usage",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage",
- "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
+ "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
},
{
"objectID": "table-layout.html#keep-and-drop-variables",
"href": "table-layout.html#keep-and-drop-variables",
"title": "Regression Tables via pf.etable()",
"section": "Keep and drop variables",
- "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#hide-fixed-effects-or-se-type-rows",
@@ -1456,49 +1456,49 @@
"href": "table-layout.html#display-p-values-or-confidence-intervals",
"title": "Regression Tables via pf.etable()",
"section": "Display p-values or confidence intervals",
- "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#significance-levels-and-rounding",
"href": "table-layout.html#significance-levels-and-rounding",
"title": "Regression Tables via pf.etable()",
"section": "Significance levels and rounding",
- "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
+ "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
},
{
"objectID": "table-layout.html#other-output-formats",
"href": "table-layout.html#other-output-formats",
"title": "Regression Tables via pf.etable()",
"section": "Other output formats",
- "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf2 - x x - x x\nf1 x x x x x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
+ "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf1 x x x x x x\nf2 - x x - x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
},
{
"objectID": "table-layout.html#rename-variables",
"href": "table-layout.html#rename-variables",
"title": "Regression Tables via pf.etable()",
"section": "Rename variables",
- "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-model-headlines",
"href": "table-layout.html#custom-model-headlines",
"title": "Regression Tables via pf.etable()",
"section": "Custom model headlines",
- "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
+ "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
},
{
"objectID": "table-layout.html#further-custom-model-information",
"href": "table-layout.html#further-custom-model-information",
"title": "Regression Tables via pf.etable()",
"section": "Further custom model information",
- "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-table-notes",
"href": "table-layout.html#custom-table-notes",
"title": "Regression Tables via pf.etable()",
"section": "Custom table notes",
- "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
+ "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
},
{
"objectID": "table-layout.html#publication-ready-latex-tables",
@@ -1519,49 +1519,49 @@
"href": "table-layout.html#summarize-by-characteristics-in-columns-and-rows",
"title": "Regression Tables via pf.etable()",
"section": "Summarize by characteristics in columns and rows",
- "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.19\n 2.35\n -0.04\n 2.21\n -0.05\n 2.28\n -0.23\n 2.39\n \n \n Wealth\n -0.72\n 5.71\n 0.05\n 4.99\n -0.09\n 5.53\n -0.56\n 6.06\n \n \n Age\n 1.07\n 0.81\n 0.96\n 0.80\n 1.08\n 0.81\n 1.06\n 0.81\n \n \n Years of Schooling\n 0.05\n 3.10\n -0.28\n 2.86\n -0.18\n 3.13\n -0.09\n 3.08\n \n \n nobs\n \n \n Number of observations\n 229\n \n 244\n \n 270\n \n 254\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.19(2.35)\n -0.04(2.21)\n -0.05(2.28)\n -0.23(2.39)\n \n \n Wealth\n -0.72(5.71)\n 0.05(4.99)\n -0.09(5.53)\n -0.56(6.06)\n \n \n Age\n 1.07(0.81)\n 0.96(0.80)\n 1.08(0.81)\n 1.06(0.81)\n \n \n Years of Schooling\n 0.05(3.10)\n -0.28(2.86)\n -0.18(3.13)\n -0.09(3.08)\n \n \n nobs\n \n \n Number of observations\n 229\n 244\n 270\n 254\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 229\n -0.19\n 2.35\n 270\n -0.05\n 2.28\n \n \n Wealth\n 229\n -0.72\n 5.71\n 270\n -0.09\n 5.53\n \n \n Age\n 229\n 1.07\n 0.81\n 270\n 1.08\n 0.81\n \n \n Years of Schooling\n 229\n 0.05\n 3.10\n 270\n -0.18\n 3.13\n \n \n White collar\n \n \n Wage\n 244\n -0.04\n 2.21\n 254\n -0.23\n 2.39\n \n \n Wealth\n 244\n 0.05\n 4.99\n 254\n -0.56\n 6.06\n \n \n Age\n 244\n 0.96\n 0.80\n 254\n 1.06\n 0.81\n \n \n Years of Schooling\n 244\n -0.28\n 2.86\n 254\n -0.09\n 3.08\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
+ "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.18\n 2.35\n 0.10\n 2.26\n -0.28\n 2.33\n -0.13\n 2.28\n \n \n Wealth\n -0.33\n 5.51\n -0.10\n 5.54\n -0.39\n 5.48\n -0.45\n 5.83\n \n \n Age\n 1.05\n 0.82\n 1.05\n 0.81\n 1.02\n 0.81\n 1.05\n 0.80\n \n \n Years of Schooling\n -0.14\n 2.99\n -0.31\n 2.92\n 0.12\n 3.24\n -0.19\n 3.02\n \n \n nobs\n \n \n Number of observations\n 254\n \n 242\n \n 246\n \n 255\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.18(2.35)\n 0.10(2.26)\n -0.28(2.33)\n -0.13(2.28)\n \n \n Wealth\n -0.33(5.51)\n -0.10(5.54)\n -0.39(5.48)\n -0.45(5.83)\n \n \n Age\n 1.05(0.82)\n 1.05(0.81)\n 1.02(0.81)\n 1.05(0.80)\n \n \n Years of Schooling\n -0.14(2.99)\n -0.31(2.92)\n 0.12(3.24)\n -0.19(3.02)\n \n \n nobs\n \n \n Number of observations\n 254\n 242\n 246\n 255\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 254\n -0.18\n 2.35\n 246\n -0.28\n 2.33\n \n \n Wealth\n 254\n -0.33\n 5.51\n 246\n -0.39\n 5.48\n \n \n Age\n 254\n 1.05\n 0.82\n 246\n 1.02\n 0.81\n \n \n Years of Schooling\n 254\n -0.14\n 2.99\n 246\n 0.12\n 3.24\n \n \n White collar\n \n \n Wage\n 242\n 0.10\n 2.26\n 255\n -0.13\n 2.28\n \n \n Wealth\n 242\n -0.10\n 5.54\n 255\n -0.45\n 5.83\n \n \n Age\n 242\n 1.05\n 0.81\n 255\n 1.05\n 0.80\n \n \n Years of Schooling\n 242\n -0.31\n 2.92\n 255\n -0.19\n 3.02\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
},
{
"objectID": "table-layout.html#basic-usage-of-make_table",
"href": "table-layout.html#basic-usage-of-make_table",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage of make_table",
- "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n -0.04\n -1.77\n -1.68\n 1.89\n \n \n 1\n 0.39\n 1.05\n -0.07\n -1.59\n \n \n 2\n 1.1\n -0.18\n 0.3\n -1.25\n \n \n 3\n 1.78\n -0.18\n -0.09\n -0.08\n \n\n \n \n \n These are notes"
+ "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n 0.87\n -1.25\n -1.78\n 1.12\n \n \n 1\n -0.88\n 1.08\n -0.47\n -0.5\n \n \n 2\n -0.31\n 1.04\n 0.56\n 0.12\n \n \n 3\n 0.37\n -0.63\n -0.72\n 1.4\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#mutiindex-dataframes",
"href": "table-layout.html#mutiindex-dataframes",
"title": "Regression Tables via pf.etable()",
"section": "Mutiindex DataFrames",
- "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes"
+ "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#example-styling",
"href": "table-layout.html#example-styling",
"title": "Regression Tables via pf.etable()",
"section": "Example Styling",
- "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#defining-table-styles-some-examples",
"href": "table-layout.html#defining-table-styles-some-examples",
"title": "Regression Tables via pf.etable()",
"section": "Defining Table Styles: Some Examples",
- "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "replicating-the-effect.html",
"href": "replicating-the-effect.html",
"title": "Replicating Examples from “The Effect”",
"section": "",
- "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\ncausaldata: 0.1.4\npyfixest : 0.24.3"
+ "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\ncausaldata: 0.1.4"
},
{
"objectID": "replicating-the-effect.html#chapter-4-describing-relationships",
"href": "replicating-the-effect.html#chapter-4-describing-relationships",
"title": "Replicating Examples from “The Effect”",
"section": "Chapter 4: Describing Relationships",
- "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
+ "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
},
{
"objectID": "replicating-the-effect.html#chapter-13-regression",
@@ -1624,7 +1624,7 @@
"href": "difference-in-differences.html#inspecting-the-outcome-variable",
"title": "Difference-in-Differences Estimation",
"section": "Inspecting the Outcome Variable",
- "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)"
+ "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n title = \"Outcome Plot\"\n)"
},
{
"objectID": "difference-in-differences.html#one-shot-adoption-static-and-dynamic-specifications",
diff --git a/table-layout.html b/table-layout.html
index 3dd0c424..2773792b 100644
--- a/table-layout.html
+++ b/table-layout.html
@@ -245,7 +245,7 @@ Regression Tables via pf.etable()
Table Layout with PyFixest
Pyfixest comes with functions to generate publication-ready tables. Regression tables are generated with pf.etable()
, which can output different formats, for instance using the Great Tables package or generating formatted LaTex Tables using booktabs. There are also further functions pf.dtable()
to display descriptive statistics and pf.make_table()
generating formatted tables from pandas dataframes in the same layout.
To begin, we load some libraries and fit a set of regression models.
-
+
import numpy as np
import pandas as pd
import pylatex as pl # for the latex table; note: not a dependency of pyfixest - needs manual installation
@@ -267,7 +267,7 @@ Table Layout wi
= pf.feols("Y2 ~ X1 *X2 | f1 + f2", data=data) fit6
-
+
@@ -301,7 +301,7 @@ Table Layout wi
-
+
@@ -338,54 +338,54 @@ Table Layout wi
Basic Usage
We can compare all regression models via the pyfixest-internal pf.etable()
function:
-
+
pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
-
+
@@ -444,20 +444,20 @@ Basic Usage
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -507,54 +507,54 @@ Basic Usage
You can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:
-
+
"Y+Y2~csw(X1,X2,X1:X2)", data=data)) pf.etable(pf.feols(
-
+
@@ -667,54 +667,54 @@ Basic Usage
Keep and drop variables
etable
allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.
-
+
="X1") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -764,20 +764,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -827,54 +827,54 @@ Keep and drop vari
We can use the exact_match
argument to select a specific set of variables:
-
+
=["X1", "X2"], exact_match=True) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -924,20 +924,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -987,54 +987,54 @@ Keep and drop vari
We can also easily drop variables via the drop
argument:
-
+
=["X1"]) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop
-
+
@@ -1075,20 +1075,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1141,54 +1141,54 @@ Keep and drop vari
Hide fixed effects or SE-type rows
We can hide the rows showing the relevant fixed effects and those showing the S.E. type by setting show_fe=False
and show_setype=False
(for instance when the set of fixed effects or the estimation method for the std. errors is the same for all models and you want to describe this in the text or table notes rather than displaying it in the table).
-
+
=False, show_se_type=False) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], show_fe
-
+
@@ -1283,54 +1283,54 @@ Hide fi
Display p-values or confidence intervals
By default, pf.etable()
reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt
function argument.
-
+
="b \n (se) \n [p]") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt
-
+
@@ -1389,20 +1389,20 @@ D
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1455,54 +1455,54 @@ D
Significance levels and rounding
Additionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code
and digits
function arguments:
-
+
=[0.01, 0.05, 0.1], digits=5) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code
-
+
@@ -1561,20 +1561,20 @@ Significa
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1627,7 +1627,7 @@ Significa
Other output formats
By default, pf.etable()
returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type
argument.
-
+
# Pandas styler output:
pf.etable(
@@ -1689,20 +1689,20 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
-0.04082 (0.08093)
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1738,7 +1738,7 @@ Other output formats<
-
+
# Markdown output:
pf.etable(
@@ -1758,8 +1758,8 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
X1:X2 0.01057 -0.04082
(0.01818) (0.08093)
------------------------------------------------------------------------------------------------
-f2 - x x - x x
f1 x x x x x x
+f2 - x x - x x
------------------------------------------------------------------------------------------------
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
@@ -1769,7 +1769,7 @@ Other output formats<
To obtain latex output use format = "tex"
. If you want to save the table as a tex file, you can use the filename=
argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True
argument. Etable will use latex packages booktabs
, threeparttable
and makecell
for the table layout, so don’t forget to include these packages in your latex document.
-
+
# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):
= pf.etable(
tab
@@ -1780,7 +1780,7 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
)
The following code generates a pdf including the regression table which you can display clicking on the link below the cell:
-
+
## Use pylatex to create a tex file with the table
@@ -1814,7 +1814,7 @@ Other output formats<
Rename variables
You can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels
argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).
-
+
Poisson Regression
-= pf.get_data(model="Fepois") data
= pf.fepois(fml="Y ~ X1 + X2 | f1 + f2", data=data, vcov="iid", iwls_tol=1e-10)
fit_iid = pf.fepois(
fit_hetero ="Y ~ X1 + X2 | f1 + f2", data=data, vcov="hetero", iwls_tol=1e-10
@@ -1065,21 +1065,21 @@ fmlPoisson Regression
- stats.vcov(fit_r_iid) fit_iid._vcov
array([[ 1.20791284e-08, -6.55604931e-10],
[-6.55604931e-10, 1.69958097e-09]])
- stats.vcov(fit_r_hetero) fit_hetero._vcov
array([[ 2.18101847e-08, -7.38711972e-10],
[-7.38711972e-10, 3.07587753e-09]])
- stats.vcov(fit_r_crv) fit_crv._vcov
array([[ 1.58300904e-08, -1.20806815e-10],
@@ -1087,7 +1087,7 @@ Poisson Regression
We conclude by comparing all estimation results via the tidy
methods:
fit_iid.tidy()
Poisson Regression
pd.DataFrame(broom.tidy_fixest(fit_r_iid)).T
Poisson Regression
fit_hetero.tidy()
Poisson Regression
pd.DataFrame(broom.tidy_fixest(fit_r_hetero)).T
Poisson Regression
fit_crv.tidy()
Poisson Regression
pd.DataFrame(broom.tidy_fixest(fit_r_crv)).T
Difference-in-Differences Estimation
See also NBER SI methods lectures on Linear Panel Event Studies.
Setup
-from importlib import resources
import pandas as pd
@@ -272,7 +272,7 @@ Setup
%autoreload 2
Setup
Setup
# one-shot adoption data - parallel trends is true
= get_sharkfin()
df_one_cohort df_one_cohort.head()
Setup
# multi-cohort adoption data
= pd.read_csv(
df_multi_cohort "pyfixest.did.data").joinpath("df_het.csv")
@@ -536,7 +536,7 @@ resources.files(Setup
Examining Treatment Timing
Before any DiD estimation, we need to examine the treatment timing, since it is crucial to our choice of estimator.
-
+
pf.panelview(
df_one_cohort,="unit",
@@ -557,7 +557,7 @@ unitExamining Treat
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -580,7 +580,7 @@ unitExamining Treat
We immediately see that we have staggered adoption of treatment in the second case, which implies that a naive application of 2WFE might yield biased estimates under substantial effect heterogeneity.
We can also plot treatment assignment in a disaggregated fashion, which gives us a sense of cohort sizes.
-
+
pf.panelview(
df_multi_cohort,="unit",
@@ -604,7 +604,7 @@ unitExamining Treat
Inspecting the Outcome Variable
pf.panelview()
further allows us to inspect the “outcome” variable over time:
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -612,20 +612,19 @@ outcomeInspecting
="year",
time="treat",
treat=True,
- collapse_to_cohort=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
We immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.
We can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.
-
+
pf.panelview(
df_multi_cohort,="dep_var",
@@ -633,13 +632,12 @@ outcomeInspecting
="year",
time="treat",
treat=100,
- subsamp=(2.5, 0.8),
- figsize= "Outcome Plot"
- title )
+= "Outcome Plot"
+ title )
@@ -648,7 +646,7 @@ Inspecting
One-shot adoption: Static and Dynamic Specifications
After taking a first look at the data, let’s turn to estimation. We return to the df_one_cohort
data set (without staggered treatment rollout).
-
+
= pf.feols(
fit_static_twfe "Y ~ treat | unit + year",
@@ -671,14 +669,14 @@ df_one_cohort,
+
= pf.feols(
fit_dynamic_twfe "Y ~ i(year, ever_treated, ref = 14) | unit + year",
df_one_cohort,={"CRV1": "unit"},
vcov )
-
+
fit_dynamic_twfe.iplot(=False,
coord_flip="Event Study",
@@ -688,7 +686,7 @@ title=rename_event_study_coefs(fit_dynamic_twfe._coefnames),
)
labels
-
+
-
+
fit_lpdid.iplot(=False,
coord_flip="Local-Projections-Estimator",
@@ -1167,7 +1165,7 @@ titleLocal Project
=18.5,
xintercept ).show()
-
+
@@ -297,7 +297,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
-
+
@@ -390,7 +390,7 @@ Marginal Effects and Hypothesis Tests via marginaleffect
Suppose we were interested in testing the hypothesis that \(X_{1} = X_{2}\). Given the relatively large differences in coefficients and small standard errors, we will likely reject the null that the two parameters are equal.
We can run the formal test via the hypotheses
function from the marginaleffects
package.
-
+
"X1 - X2 = 0") hypotheses(fit,
@@ -545,7 +545,7 @@ PyFixest 0.18.0
Additionally, model_matrix_fixest
now returns a dictionary instead of a tuple.
Brings back fixed effects reference setting via i(var1, var2, ref)
syntax. Deprecates the i_ref1
, i_ref2
function arguments. I.e. it is again possible to e.g. run
-
+
import pyfixest as pf
= pf.get_data()
data
@@ -553,7 +553,7 @@ PyFixest 0.18.0
0:8] fit1.coef()[
Via the ref
syntax, via can set the reference level:
-
+
= pf.feols("Y ~ i(f1, X2, ref = 1)", data=data)
fit2 0:8] fit2.coef()[
@@ -562,7 +562,7 @@ PyFixest 0.18.0
PyFixest 0.17.0
Restructures the codebase and reorganizes how users can interact with the pyfixest
API. It is now recommended to use pyfixest
in the following way:
-
+
import numpy as np
import pyfixest as pf
= pf.get_data()
@@ -630,7 +630,7 @@ data PyFixest 0.17.0
The update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!
Adds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!
-
+
= True) fit.confint(joint
@@ -647,18 +647,18 @@ PyFixest 0.17.0
Intercept
-0.379125
-1.178573
+0.373326
+1.184372
D
--1.759996
--1.045238
+-1.765180
+-1.040053
f1
--0.014143
-0.023692
+-0.014418
+0.023966
@@ -667,7 +667,7 @@ PyFixest 0.17.0
Adds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv()
method.
-
+
= "D", cluster = "group_id") fit.ccv(treatment
/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.
@@ -693,11 +693,11 @@ PyFixest 0.17.0
CCV
-1.4026168622179929
-0.301475
--4.652514
-0.000198
--2.035992
--0.769241
+0.275208
+-5.096563
+0.000075
+-1.980808
+-0.824425
CRV1
@@ -739,7 +739,7 @@ PyFixest 0.14.0
- Changes all docstrings to
numpy
format.
- Difference-in-differences estimation functions now need to be imported via the
pyfixest.did.estimation
module:
-
+
from pyfixest.did.estimation import did2s, lpdid, event_study
diff --git a/quarto_example/QuartoExample.pdf b/quarto_example/QuartoExample.pdf
index 78921f36..41692775 100644
Binary files a/quarto_example/QuartoExample.pdf and b/quarto_example/QuartoExample.pdf differ
diff --git a/quickstart.html b/quickstart.html
index 5bd8b2d2..7fca5546 100644
--- a/quickstart.html
+++ b/quickstart.html
@@ -281,7 +281,7 @@ What is a fix
Read Sample Data
In a first step, we load the module and some synthetic example data:
-
+
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
@@ -302,7 +302,7 @@ Read Sample Data
data.head()
-
+
@@ -336,7 +336,7 @@ Read Sample Data
-
+
@@ -370,7 +370,7 @@ Read Sample Data
-
+
-pandas : 2.2.3
-matplotlib: 3.9.2
-pyfixest : 0.24.3
+pyfixest : 0.24.3
numpy : 1.26.4
+matplotlib: 3.9.2
+pandas : 2.2.3
@@ -507,7 +507,7 @@ Read Sample Data
-
+
data.info()
<class 'pandas.core.frame.DataFrame'>
@@ -535,7 +535,7 @@ Read Sample Data
OLS Estimation
We are interested in the relation between the dependent variable Y
and the independent variables X1
using a fixed effect model for group_id
. Let’s see how the data looks like:
-
+
= data.plot(kind="scatter", x="X1", y="Y", c="group_id", colormap="viridis") ax
@@ -546,7 +546,7 @@ OLS Estimation
We can estimate a fixed effects regression via the feols()
function. feols()
has three arguments: a two-sided model formula, the data, and optionally, the type of inference.
-
+
= pf.feols(fml="Y ~ X1 | group_id", data=data, vcov="HC1")
fit type(fit)
@@ -559,7 +559,7 @@ OLS Estimation
Inspecting Model Results
To inspect the results, we can use a summary function or method:
-
+
fit.summary()
###
@@ -577,54 +577,54 @@ Inspecting Model
Or display a formatted regression table:
-
+
pf.etable(fit)
-
+
@@ -687,7 +687,7 @@ Inspecting Model
Alternatively, the .summarize
module contains a summary
function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable()
, please take a look at the dedicated vignette.
-
+
pf.summary(fit)
###
@@ -705,7 +705,7 @@ Inspecting Model
You can access individual elements of the summary via dedicated methods: .tidy()
returns a “tidy” pd.DataFrame
, .coef()
returns estimated parameters, and se()
estimated standard errors. Other methods include pvalue()
, confint()
and tstat()
.
-
+
fit.tidy()
@@ -748,7 +748,7 @@ Inspecting Model
-
+
fit.coef()
Coefficient
@@ -756,7 +756,7 @@ Inspecting Model
Name: Estimate, dtype: float64
-
+
fit.se()
Coefficient
@@ -764,7 +764,7 @@ Inspecting Model
Name: Std. Error, dtype: float64
-
+
fit.tstat()
Coefficient
@@ -772,7 +772,7 @@ Inspecting Model
Name: t value, dtype: float64
-
+
fit.confint()
@@ -799,11 +799,11 @@ Inspecting Model
Last, model results can be visualized via dedicated methods for plotting:
-
+
fit.coefplot()# or pf.coefplot([fit])
-
+
@@ -522,7 +522,7 @@ Examples
-
+
@@ -671,7 +671,7 @@ Examples
In a first step, we estimate a classical event study model:
-
+
# estimate the model
= pf.did2s(
fit
@@ -761,10 +761,10 @@ df_het,Examples
We can also inspect the model visually:
-
+
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -545,7 +545,7 @@ Examples
-
+
diff --git a/reference/did.estimation.lpdid.html b/reference/did.estimation.lpdid.html
index 6d7187ce..b74e15d2 100644
--- a/reference/did.estimation.lpdid.html
+++ b/reference/did.estimation.lpdid.html
@@ -505,7 +505,7 @@ Returns
Examples
-
+
import pandas as pd
import pyfixest as pf
@@ -528,7 +528,7 @@ Examples
= [1200, 400], coord_flip=False).show() fit.iplot(figsize
-
+
@@ -562,7 +562,7 @@ Examples
-
+
-
+
@@ -588,7 +588,7 @@ Examples
-
+
@@ -638,7 +638,7 @@ Examples
Calling feols()
returns an instance of the [Feols(/reference/Feols.qmd) class. The summary()
method can be used to print the results.
An alternative way to retrieve model results is via the tidy()
method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.
-
+
fit.tidy()
@@ -692,17 +692,17 @@ Examples
You can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef()
for the coefficients, fit.se()
for the standard errors, fit.tstat()
for the t-statistics, and fit.pval()
for the p-values, and fit.confint()
for the confidence intervals.
The employed type of inference can be specified via the vcov
argument. If vcov is not provided, PyFixest
employs the fixest
default of iid inference, unless there are fixed effects in the model, in which case feols()
clusters the standard error by the first fixed effect (CRV1 inference).
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="iid")
fit1 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov="hetero")
fit2 = pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1"}) fit3
Supported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {"CRV1": "f1"}
for CRV1 inference with clustering by f1
or {"CRV3": "f1"}
for CRV3 inference with clustering by f1
. For two-way clustering, you can provide a formula string, e.g. {"CRV1": "f1 + f2"}
for CRV1 inference with clustering by f1
.
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data, vcov={"CRV1": "f1 + f2"}) fit4
Inference can be adjusted post estimation via the vcov
method:
-
+
fit.summary()"iid").summary() fit.vcov(
@@ -736,7 +736,7 @@ Examples
The ssc
argument specifies the small sample correction for inference. In general, feols()
uses all of fixest::feols()
defaults, but sets the fixef.K
argument to "none"
whereas the fixest::feols()
default is "nested"
. See here for more details: link to github.
feols()
supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1
and one with fixed effects for f2
using the sw()
syntax.
-
+
= pf.feols("Y ~ X1 + X2 | sw(f1, f2)", data)
fit type(fit)
@@ -744,58 +744,58 @@ Examples
The returned object is an instance of the FixestMulti
class. You can access the results of the first model via fit.fetch_model(0)
and the results of the second model via fit.fetch_model(1)
. You can compare the model results via the etable()
function:
-
+
0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
Model: Y~X1+X2|f1
Model: Y~X1+X2|f2
-
+
@@ -837,14 +837,14 @@ Examples
fe
-f1
-x
+f2
-
+x
-f2
--
+f1
x
+-
stats
@@ -878,7 +878,7 @@ Examples
Other supported multiple estimation syntax include sw0()
, csw()
and csw0()
. While sw()
adds variables in a “stepwise” fashion, csw()
does so cumulatively.
-
+
= pf.feols("Y ~ X1 + X2 | csw(f1, f2)", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -886,51 +886,51 @@ Examples
Model: Y~X1+X2|f1+f2
-
+
@@ -972,13 +972,13 @@ Examples
fe
-f1
-x
+f2
+-
x
-f2
--
+f1
+x
x
@@ -1013,7 +1013,7 @@ Examples
The sw0()
and csw0()
syntax are similar to sw()
and csw()
, but start with a model that excludes the variables specified in sw()
and csw()
:
-
+
= pf.feols("Y ~ X1 + X2 | sw0(f1, f2)", data)
fit 0), fit.fetch_model(1), fit.fetch_model(2)]) pf.etable([fit.fetch_model(
@@ -1022,51 +1022,51 @@ Examples
Model: Y~X1+X2|f2
-
+
@@ -1121,16 +1121,16 @@ Examples
fe
-f1
+f2
-
-x
-
+x
-f2
--
+f1
-
x
+-
stats
@@ -1167,7 +1167,7 @@ Examples
The feols()
function also supports multiple dependent variables. The following example estimates two models, one with Y1
as the dependent variable and one with Y2
as the dependent variable.
-
+
= pf.feols("Y + Y2 ~ X1 | f1 + f2", data)
fit 0), fit.fetch_model(1)]) pf.etable([fit.fetch_model(
@@ -1175,51 +1175,51 @@ Examples
Model: Y2~X1|f1+f2
-
+
@@ -1255,12 +1255,12 @@ Examples
fe
-f1
+f2
x
x
-f2
+f1
x
x
@@ -1296,7 +1296,7 @@ Examples
It is possible to combine different multiple estimation operators:
-
+
= pf.feols("Y + Y2 ~ X1 | sw(f1, f2)", data)
fit 0),
pf.etable([fit.fetch_model(1),
@@ -1311,51 +1311,51 @@ fit.fetch_model(Examples
Model: Y2~X1|f2
-
+
@@ -1401,18 +1401,18 @@ Examples
fe
-f1
-x
-x
+f2
-
-
+x
+x
-f2
--
--
+f1
x
x
+-
+-
stats
@@ -1453,7 +1453,7 @@ Examples
In general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols()
implements a caching mechanism that stores the demeaned covariates.
Besides OLS, feols()
also supports IV estimation via three part formulas:
-
+
= pf.feols("Y ~ X2 | f1 + f2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1507,7 +1507,7 @@ Examples
Here, X1
is the endogenous variable and Z1
is the instrument. f1
and f2
are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:
-
+
= pf.feols("Y ~ X2 | X1 ~ Z1", data)
fit fit.tidy()
@@ -1571,7 +1571,7 @@ Examples
Last, feols()
supports interaction of variables via the i()
syntax. Documentation on this is tba.
After fitting a model via feols()
, you can use the predict()
method to get the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit 0:5] fit.predict()[
@@ -1579,7 +1579,7 @@ Examples
The predict()
method also supports a newdata
argument to predict on new data, which returns a numpy array of the predicted values:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit =data)[0:5] fit.predict(newdata
@@ -1587,11 +1587,11 @@ Examples
Last, you can plot the results of a model via the coefplot()
method:
-
+
= pf.feols("Y ~ X1 + X2 | f1 + f2", data)
fit fit.coefplot()
-
+
@@ -569,7 +569,7 @@ Examples
-
+
diff --git a/reference/report.coefplot.html b/reference/report.coefplot.html
index d8796924..098ac83c 100644
--- a/reference/report.coefplot.html
+++ b/reference/report.coefplot.html
@@ -528,7 +528,7 @@ Returns
Examples
-
+
import pyfixest as pf
from pyfixest.report.utils import rename_categoricals
@@ -544,7 +544,7 @@ Examples
= "both") pf.coefplot([fit1], joint
-
+
@@ -578,7 +578,7 @@ Examples
-
+
-
+
@@ -576,7 +576,7 @@ Examples
-
+
-
+
@@ -497,7 +497,7 @@ Examples
-
+
diff --git a/replicating-the-effect.html b/replicating-the-effect.html
index 804007ca..a6a6a321 100644
--- a/replicating-the-effect.html
+++ b/replicating-the-effect.html
@@ -234,7 +234,7 @@ Replicating Examples from “The Effect”
This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.
-
+
from causaldata import Mroz, gapminder, organ_donations, restaurant_inspections
import pyfixest as pf
@@ -243,7 +243,7 @@ Replicating Examples from “The Effect”
%watermark --iversions
-
+
@@ -277,7 +277,7 @@ Replicating Examples from “The Effect”
-
+
-causaldata: 0.1.4
-pyfixest : 0.24.3
+pyfixest : 0.24.3
+causaldata: 0.1.4
Chapter 4: Describing Relationships
-
+
# Read in data
= Mroz.load_pandas().data
dt # Keep just working women
@@ -329,7 +329,7 @@ Chapter
= pf.feols(fml="lwg ~ csw(inc, wc, k5)", data=dt, vcov="iid")
fit pf.etable(fit)
-/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning:
+/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning:
A value is trying to be set on a copy of a slice from a DataFrame.
Try using .loc[row_indexer,col_indexer] = value instead
@@ -337,51 +337,51 @@ Chapter
dt.loc[:, "earn"] = dt["lwg"].apply("exp")
-
+
@@ -479,7 +479,7 @@ Chapter
Chapter 13: Regression
Example 1
-
+
= restaurant_inspections.load_pandas().data
res = res.inspection_score.astype(float)
res.inspection_score = res.NumberofLocations.astype(float)
@@ -488,51 +488,51 @@ res.NumberofLocations Example 1
= pf.feols(fml="inspection_score ~ NumberofLocations", data=res)
fit pf.etable([fit])
-
+
@@ -595,7 +595,7 @@ Example 1
Example 2
-
+
= restaurant_inspections.load_pandas().data
df
= pf.feols(
@@ -605,51 +605,51 @@ fit1 Example 2
pf.etable([fit1, fit2])
-
+
@@ -746,7 +746,7 @@ Example 2
Example 3: HC Standard Errors
-
+
="inspection_score ~ Year + Weekend", data=df, vcov="HC3").summary() pf.feols(fml
###
@@ -768,7 +768,7 @@ Example 3: HC
Example 4: Clustered Standard Errors
-
+
pf.feols(="inspection_score ~ Year + Weekend", data=df, vcov={"CRV1": "Weekend"}
fml ).tidy()
@@ -834,7 +834,7 @@ Exampl
Example 5: Bootstrap Inference
-
+
= pf.feols(fml="inspection_score ~ Year + Weekend", data=df)
fit =999, param="Year") fit.wildboottest(reps
@@ -857,7 +857,7 @@ Example 1
Example 2
-
+
= gapminder.load_pandas().data
gm "logGDPpercap"] = gm["gdpPercap"].apply("log")
gm[
@@ -943,7 +943,7 @@ Example 2
Example 3: TWFE
-
+
# Set our individual and time (index) for our data
= pf.feols(fml="lifeExp ~ np.log(gdpPercap) | country + year", data=gm)
fit fit.summary()
@@ -968,7 +968,7 @@ Example 3: TWFE
Chapter 18: Difference-in-Differences
Example 1
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
@@ -996,7 +996,7 @@ Example 1
Example 3: Dynamic Treatment Effect
-
+
= organ_donations.load_pandas().data
od
# Create Treatment Variable
diff --git a/search.json b/search.json
index f717fbdc..f69e7216 100644
--- a/search.json
+++ b/search.json
@@ -74,7 +74,7 @@
"href": "news.html#pyfixest-0.17.0",
"title": "News",
"section": "PyFixest 0.17.0",
- "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.379125\n1.178573\n\n\nD\n-1.759996\n-1.045238\n\n\nf1\n-0.014143\n0.023692\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.301475\n-4.652514\n0.000198\n-2.035992\n-0.769241\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
+ "text": "PyFixest 0.17.0\n\nRestructures the codebase and reorganizes how users can interact with the pyfixest API. It is now recommended to use pyfixest in the following way:\n\nimport numpy as np\nimport pyfixest as pf\ndata = pf.get_data()\ndata[\"D\"] = data[\"X1\"] > 0\nfit = pf.feols(\"Y ~ D + f1\", data = data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.778849\n0.170261\n4.574437\n0.000005\n0.444737\n1.112961\n\n\nD\n-1.402617\n0.152224\n-9.214140\n0.000000\n-1.701335\n-1.103899\n\n\nf1\n0.004774\n0.008058\n0.592508\n0.553645\n-0.011038\n0.020587\n\n\n\n\n\n\n\nThe update should not inroduce any breaking changes. Thanks to @Wenzhi-Ding for the PR!\nAdds support for simultaneous confidence intervals via a multiplier bootstrap. Thanks to @apoorvalal for the contribution!\n\nfit.confint(joint = True)\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n0.373326\n1.184372\n\n\nD\n-1.765180\n-1.040053\n\n\nf1\n-0.014418\n0.023966\n\n\n\n\n\n\n\nAdds support for the causal cluster variance estimator by Abadie et al. (QJE, 2023) for OLS via the .ccv() method.\n\nfit.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n-1.4026168622179929\n0.275208\n-5.096563\n0.000075\n-1.980808\n-0.824425\n\n\nCRV1\n-1.402617\n0.205132\n-6.837621\n0.000002\n-1.833584\n-0.97165"
},
{
"objectID": "news.html#pyfixest-0.16.0",
@@ -298,7 +298,7 @@
"href": "quickstart.html",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "A fixed effect model is a statistical model that includes fixed effects, which are parameters that are estimated to be constant across different groups.\nExample [Panel Data]: In the context of panel data, fixed effects are parameters that are constant across different individuals or time. The typical model example is given by the following equation:\n\\[\nY_{it} = \\beta X_{it} + \\alpha_i + \\psi_t + \\varepsilon_{it}\n\\]\nwhere \\(Y_{it}\\) is the dependent variable for individual \\(i\\) at time \\(t\\), \\(X_{it}\\) is the independent variable, \\(\\beta\\) is the coefficient of the independent variable, \\(\\alpha_i\\) is the individual fixed effect, \\(\\psi_t\\) is the time fixed effect, and \\(\\varepsilon_{it}\\) is the error term. The individual fixed effect \\(\\alpha_i\\) is a parameter that is constant across time for each individual, while the time fixed effect \\(\\psi_t\\) is a parameter that is constant across individuals for each time period.\nNote however that, despite the fact that fixed effects are commonly used in panel setting, one does not need a panel data set to work with fixed effects. For example, cluster randomized trials with cluster fixed effects, or wage regressions with worker and firm fixed effects.\nIn this “quick start” guide, we will show you how to estimate a fixed effect model using the PyFixest package. We do not go into the details of the theory behind fixed effect models, but we focus on how to estimate them using PyFixest.\n\n\n\nIn a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data.\n\n\n\nWe are interested in the relation between the dependent variable Y and the independent variables X1 using a fixed effect model for group_id. Let’s see how the data looks like:\n\nax = data.plot(kind=\"scatter\", x=\"X1\", y=\"Y\", c=\"group_id\", colormap=\"viridis\")\n\n\n\n\n\n\n\n\nWe can estimate a fixed effects regression via the feols() function. feols() has three arguments: a two-sided model formula, the data, and optionally, the type of inference.\n\nfit = pf.feols(fml=\"Y ~ X1 | group_id\", data=data, vcov=\"HC1\")\ntype(fit)\n\npyfixest.estimation.feols_.Feols\n\n\nThe first part of the formula contains the dependent variable and “regular” covariates, while the second part contains fixed effects.\nfeols() returns an instance of the Fixest class.\n\n\n\nTo inspect the results, we can use a summary function or method:\n\nfit.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nOr display a formatted regression table:\n\npf.etable(fit)\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n\n\nfe\n\n\ngroup_id\nx\n\n\nstats\n\n\nObservations\n998\n\n\nS.E. type\nhetero\n\n\nR2\n0.137\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nAlternatively, the .summarize module contains a summary function, which can be applied on instances of regression model objects or lists of regression model objects. For details on how to customize etable(), please take a look at the dedicated vignette.\n\npf.summary(fit)\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: group_id\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -1.019 | 0.082 | -12.352 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.137 R2 Within: 0.126 \n\n\nYou can access individual elements of the summary via dedicated methods: .tidy() returns a “tidy” pd.DataFrame, .coef() returns estimated parameters, and se() estimated standard errors. Other methods include pvalue(), confint() and tstat().\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.019009\n0.082498\n-12.351897\n0.0\n-1.180898\n-0.857119\n\n\n\n\n\n\n\n\nfit.coef()\n\nCoefficient\nX1 -1.019009\nName: Estimate, dtype: float64\n\n\n\nfit.se()\n\nCoefficient\nX1 0.082498\nName: Std. Error, dtype: float64\n\n\n\nfit.tstat()\n\nCoefficient\nX1 -12.351897\nName: t value, dtype: float64\n\n\n\nfit.confint()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nX1\n-1.180898\n-0.857119\n\n\n\n\n\n\n\nLast, model results can be visualized via dedicated methods for plotting:\n\nfit.coefplot()\n# or pf.coefplot([fit])\n\n \n \n\n\n\n\n\nLet’s have a quick d-tour on the intuition behind fixed effects models using the example above. To do so, let us begin by comparing it with a simple OLS model.\n\nfit_simple = pf.feols(\"Y ~ X1\", data=data, vcov=\"HC1\")\n\nfit_simple.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.919 | 0.112 | 8.223 | 0.000 | 0.699 | 1.138 |\n| X1 | -1.000 | 0.082 | -12.134 | 0.000 | -1.162 | -0.838 |\n---\nRMSE: 2.158 R2: 0.123 \n\n\nWe can compare both models side by side in a regression table:\n\npf.etable([fit, fit_simple])\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-1.019***\n(0.082)\n-1.000***\n(0.082)\n\n\nIntercept\n\n0.919***\n(0.112)\n\n\nfe\n\n\ngroup_id\nx\n-\n\n\nstats\n\n\nObservations\n998\n998\n\n\nS.E. type\nhetero\nhetero\n\n\nR2\n0.137\n0.123\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nWe see that the X1 coefficient is -1.019, which is less than the value from the OLS model in column (2). Where is the difference coming from? Well, in the fixed effect model we are interested in controlling for the feature group_id. One possibility to do this is by adding a simple dummy variable for each level of group_id.\n\nfit_dummy = pf.feols(\"Y ~ X1 + C(group_id) \", data=data, vcov=\"HC1\")\n\nfit_dummy.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.760 | 0.288 | 2.640 | 0.008 | 0.195 | 1.326 |\n| X1 | -1.019 | 0.083 | -12.234 | 0.000 | -1.182 | -0.856 |\n| C(group_id)[T.1.0] | 0.380 | 0.451 | 0.844 | 0.399 | -0.504 | 1.264 |\n| C(group_id)[T.2.0] | 0.084 | 0.389 | 0.216 | 0.829 | -0.680 | 0.848 |\n| C(group_id)[T.3.0] | 0.790 | 0.415 | 1.904 | 0.057 | -0.024 | 1.604 |\n| C(group_id)[T.4.0] | -0.189 | 0.388 | -0.487 | 0.626 | -0.950 | 0.572 |\n| C(group_id)[T.5.0] | 0.537 | 0.388 | 1.385 | 0.166 | -0.224 | 1.297 |\n| C(group_id)[T.6.0] | 0.307 | 0.398 | 0.771 | 0.441 | -0.474 | 1.087 |\n| C(group_id)[T.7.0] | 0.015 | 0.422 | 0.035 | 0.972 | -0.814 | 0.844 |\n| C(group_id)[T.8.0] | 0.382 | 0.406 | 0.941 | 0.347 | -0.415 | 1.179 |\n| C(group_id)[T.9.0] | 0.219 | 0.417 | 0.526 | 0.599 | -0.599 | 1.037 |\n| C(group_id)[T.10.0] | -0.363 | 0.422 | -0.861 | 0.390 | -1.191 | 0.465 |\n| C(group_id)[T.11.0] | 0.201 | 0.387 | 0.520 | 0.603 | -0.559 | 0.961 |\n| C(group_id)[T.12.0] | -0.110 | 0.410 | -0.268 | 0.788 | -0.915 | 0.694 |\n| C(group_id)[T.13.0] | 0.126 | 0.440 | 0.287 | 0.774 | -0.736 | 0.989 |\n| C(group_id)[T.14.0] | 0.353 | 0.416 | 0.848 | 0.397 | -0.464 | 1.170 |\n| C(group_id)[T.15.0] | 0.469 | 0.398 | 1.179 | 0.239 | -0.312 | 1.249 |\n| C(group_id)[T.16.0] | -0.135 | 0.396 | -0.340 | 0.734 | -0.913 | 0.643 |\n| C(group_id)[T.17.0] | -0.005 | 0.401 | -0.013 | 0.989 | -0.792 | 0.781 |\n| C(group_id)[T.18.0] | 0.283 | 0.403 | 0.702 | 0.483 | -0.508 | 1.074 |\n---\nRMSE: 2.141 R2: 0.137 \n\n\nThis is does not scale well! Imagine you have 1000 different levels of group_id. You would need to add 1000 dummy variables to your model. This is where fixed effect models come in handy. They allow you to control for these fixed effects without adding all these dummy variables. The way to do it is by a demeaning procedure. The idea is to subtract the average value of each level of group_id from the respective observations. This way, we control for the fixed effects without adding all these dummy variables. Let’s try to do this manually:\n\ndef _demean_column(df: pd.DataFrame, column: str, by: str) -> pd.Series:\n return df[column] - df.groupby(by)[column].transform(\"mean\")\n\n\nfit_demeaned = pf.feols(\n fml=\"Y_demeaned ~ X1_demeaned\",\n data=data.assign(\n Y_demeaned=lambda df: _demean_column(df, \"Y\", \"group_id\"),\n X1_demeaned=lambda df: _demean_column(df, \"X1\", \"group_id\"),\n ),\n vcov=\"HC1\",\n)\n\nfit_demeaned.summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y_demeaned, Fixed effects: 0\nInference: HC1\nObservations: 998\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| Intercept | 0.003 | 0.068 | 0.041 | 0.968 | -0.130 | 0.136 |\n| X1_demeaned | -1.019 | 0.083 | -12.345 | 0.000 | -1.181 | -0.857 |\n---\nRMSE: 2.141 R2: 0.126 \n\n\nWe get the same results as the fixed effect model Y1 ~ X | group_id above. The PyFixest package uses a more efficient algorithm to estimate the fixed effect model, but the intuition is the same.\n\n\n\nYou can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#what-is-a-fixed-effect-model",
@@ -312,7 +312,7 @@
"href": "quickstart.html#read-sample-data",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npandas : 2.2.3\nmatplotlib: 3.9.2\npyfixest : 0.24.3\nnumpy : 1.26.4\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
+ "text": "In a first step, we load the module and some synthetic example data:\n\nimport matplotlib.pyplot as plt\nimport numpy as np\nimport pandas as pd\nfrom lets_plot import LetsPlot\n\nimport pyfixest as pf\n\nLetsPlot.setup_html()\n\nplt.style.use(\"seaborn-v0_8\")\n\n%load_ext watermark\n%config InlineBackend.figure_format = \"retina\"\n%watermark --iversions\n\ndata = pf.get_data()\n\ndata.head()\n\n\n \n \n \n\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\nnumpy : 1.26.4\nmatplotlib: 3.9.2\npandas : 2.2.3\n\n\n\n\n\n\n\n\n\n\nY\nY2\nX1\nX2\nf1\nf2\nf3\ngroup_id\nZ1\nZ2\nweights\n\n\n\n\n0\nNaN\n2.357103\n0.0\n0.457858\n15.0\n0.0\n7.0\n9.0\n-0.330607\n1.054826\n0.661478\n\n\n1\n-1.458643\n5.163147\nNaN\n-4.998406\n6.0\n21.0\n4.0\n8.0\nNaN\n-4.113690\n0.772732\n\n\n2\n0.169132\n0.751140\n2.0\n1.558480\nNaN\n1.0\n7.0\n16.0\n1.207778\n0.465282\n0.990929\n\n\n3\n3.319513\n-2.656368\n1.0\n1.560402\n1.0\n10.0\n11.0\n3.0\n2.869997\n0.467570\n0.021123\n\n\n4\n0.134420\n-1.866416\n2.0\n-3.472232\n19.0\n20.0\n6.0\n14.0\n0.835819\n-3.115669\n0.790815\n\n\n\n\n\n\n\n\ndata.info()\n\n<class 'pandas.core.frame.DataFrame'>\nRangeIndex: 1000 entries, 0 to 999\nData columns (total 11 columns):\n # Column Non-Null Count Dtype \n--- ------ -------------- ----- \n 0 Y 999 non-null float64\n 1 Y2 1000 non-null float64\n 2 X1 999 non-null float64\n 3 X2 1000 non-null float64\n 4 f1 999 non-null float64\n 5 f2 1000 non-null float64\n 6 f3 1000 non-null float64\n 7 group_id 1000 non-null float64\n 8 Z1 999 non-null float64\n 9 Z2 1000 non-null float64\n 10 weights 1000 non-null float64\ndtypes: float64(11)\nmemory usage: 86.1 KB\n\n\nWe see that some of our columns have missing data."
},
{
"objectID": "quickstart.html#ols-estimation",
@@ -340,7 +340,7 @@
"href": "quickstart.html#updating-regression-coefficients",
"title": "Getting Started with PyFixest",
"section": "",
- "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 1.1035891 , -1.12786813, -0.17762566])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 1.104781 , -1.13148511, -0.18057651])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 1.104781 , -1.13148511, -0.18057651])"
+ "text": "You can update the coefficients of a model object via the update() method, which may be useful in an online learning setting where data arrives sequentially.\nTo see this in action, let us first fit a model on a subset of the data:\n\ndata_subsample = data.sample(frac=0.5)\nm = pf.feols(\"Y ~ X1 + X2\", data=data_subsample)\n# current coefficient vector\nm._beta_hat\n\narray([ 0.89795329, -1.01756326, -0.18513421])\n\n\nThen sample 5 new observations and update the model with the new data. The update rule is\n\\[\n\\hat{\\beta}_{n+1} = \\hat{\\beta}_n + (X_{n+1}' X_{n+1})^{-1} x_{n+1} + (y_{n+1} - x_{n+1} \\hat{\\beta}_n)\n\\]\nfor a new observation \\((x_{n+1}, y_{n+1})\\).\n\nnew_points_id = np.random.choice(list(set(data.index) - set(data_subsample.index)), 5)\nX_new, y_new = (\n np.c_[np.ones(len(new_points_id)), data.loc[new_points_id][[\"X1\", \"X2\"]].values],\n data.loc[new_points_id][\"Y\"].values,\n)\nm.update(X_new, y_new)\n\narray([ 0.88420408, -1.00453996, -0.18364787])\n\n\nWe verify that we get the same results if we had estimated the model on the appended data.\n\npf.feols(\n \"Y ~ X1 + X2\", data=data.loc[data_subsample.index.append(pd.Index(new_points_id))]\n).coef().values\n\narray([ 0.88420408, -1.00453996, -0.18364787])"
},
{
"objectID": "quickstart.html#inference-via-the-wild-bootstrap",
@@ -375,7 +375,7 @@
"href": "quickstart.html#joint-confidence-intervals",
"title": "Getting Started with PyFixest",
"section": "Joint Confidence Intervals",
- "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.425936\n1.403847\n\n\nX1\n-1.160730\n-0.738152\n\n\nC(f1)[T.1.0]\n1.384234\n3.781064\n\n\nC(f1)[T.2.0]\n-2.838865\n-0.325003\n\n\nC(f1)[T.3.0]\n-1.608332\n0.983664"
+ "text": "Joint Confidence Intervals\nSimultaneous confidence bands for a vector of parameters can be computed via the joint_confint() method. See Simultaneous confidence bands: Theory, implementation, and an application to SVARs for background.\n\nfit_ci = pf.feols(\"Y ~ X1+ C(f1)\", data=data)\nfit_ci.confint(joint=True).head()\n\n\n\n\n\n\n\n\n2.5%\n97.5%\n\n\n\n\nIntercept\n-0.430485\n1.408396\n\n\nX1\n-1.161781\n-0.737102\n\n\nC(f1)[T.1.0]\n1.378276\n3.787022\n\n\nC(f1)[T.2.0]\n-2.845114\n-0.318754\n\n\nC(f1)[T.3.0]\n-1.614775\n0.990107"
},
{
"objectID": "contributing.html",
@@ -911,7 +911,7 @@
"href": "reference/estimation.estimation.feols.html#examples",
"title": "estimation.estimation.feols",
"section": "Examples",
- "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf1\nx\n-\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\n-\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf1\n-\nx\n-\n\n\nf2\n-\n-\nx\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf1\nx\nx\n\n\nf2\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf1\nx\nx\n-\n-\n\n\nf2\n-\n-\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.265821\n0.060521\n0.952408\n-0.542381\n0.574556\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
+ "text": "Examples\nAs in fixest, the [Feols(/reference/Feols.qmd) function can be used to estimate a simple linear regression model with fixed effects. The following example regresses Y on X1 and X2 with fixed effects for f1 and f2: fixed effects are specified after the | symbol.\n\nimport pyfixest as pf\n\ndata = pf.get_data()\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.summary()\n\n\n \n \n \n\n\n\n \n \n \n\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nCalling feols() returns an instance of the [Feols(/reference/Feols.qmd) class. The summary() method can be used to print the results.\nAn alternative way to retrieve model results is via the tidy() method, which returns a pandas dataframe with the estimated coefficients, standard errors, t-statistics, and p-values.\n\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-0.924046\n0.060934\n-15.164621\n2.664535e-15\n-1.048671\n-0.799421\n\n\nX2\n-0.174107\n0.014608\n-11.918277\n1.069367e-12\n-0.203985\n-0.144230\n\n\n\n\n\n\n\nYou can also access all elements in the tidy data frame by dedicated methods, e.g. fit.coef() for the coefficients, fit.se() for the standard errors, fit.tstat() for the t-statistics, and fit.pval() for the p-values, and fit.confint() for the confidence intervals.\nThe employed type of inference can be specified via the vcov argument. If vcov is not provided, PyFixest employs the fixest default of iid inference, unless there are fixed effects in the model, in which case feols() clusters the standard error by the first fixed effect (CRV1 inference).\n\nfit1 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"iid\")\nfit2 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov=\"hetero\")\nfit3 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1\"})\n\nSupported inference types are “iid”, “hetero”, “HC1”, “HC2”, “HC3”, and “CRV1”/“CRV3”. Clustered standard errors are specified via a dictionary, e.g. {\"CRV1\": \"f1\"} for CRV1 inference with clustering by f1 or {\"CRV3\": \"f1\"} for CRV3 inference with clustering by f1. For two-way clustering, you can provide a formula string, e.g. {\"CRV1\": \"f1 + f2\"} for CRV1 inference with clustering by f1.\n\nfit4 = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data, vcov={\"CRV1\": \"f1 + f2\"})\n\nInference can be adjusted post estimation via the vcov method:\n\nfit.summary()\nfit.vcov(\"iid\").summary()\n\n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: CRV1\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.061 | -15.165 | 0.000 | -1.049 | -0.799 |\n| X2 | -0.174 | 0.015 | -11.918 | 0.000 | -0.204 | -0.144 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n###\n\nEstimation: OLS\nDep. var.: Y, Fixed effects: f1+f2\nInference: iid\nObservations: 997\n\n| Coefficient | Estimate | Std. Error | t value | Pr(>|t|) | 2.5% | 97.5% |\n|:--------------|-----------:|-------------:|----------:|-----------:|-------:|--------:|\n| X1 | -0.924 | 0.054 | -16.995 | 0.000 | -1.031 | -0.817 |\n| X2 | -0.174 | 0.014 | -12.081 | 0.000 | -0.202 | -0.146 |\n---\nRMSE: 1.346 R2: 0.659 R2 Within: 0.303 \n\n\nThe ssc argument specifies the small sample correction for inference. In general, feols() uses all of fixest::feols() defaults, but sets the fixef.K argument to \"none\" whereas the fixest::feols() default is \"nested\". See here for more details: link to github.\nfeols() supports a range of multiple estimation syntax, i.e. you can estimate multiple models in one call. The following example estimates two models, one with fixed effects for f1 and one with fixed effects for f2 using the sw() syntax.\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw(f1, f2)\", data)\ntype(fit)\n\npyfixest.estimation.FixestMulti_.FixestMulti\n\n\nThe returned object is an instance of the FixestMulti class. You can access the results of the first model via fit.fetch_model(0) and the results of the second model via fit.fetch_model(1). You can compare the model results via the etable() function:\n\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\n-\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f2\n\n\nR2\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nOther supported multiple estimation syntax include sw0(), csw() and csw0(). While sw() adds variables in a “stepwise” fashion, csw() does so cumulatively.\n\nfit = pf.feols(\"Y ~ X1 + X2 | csw(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.950***\n(0.067)\n-0.924***\n(0.061)\n\n\nX2\n-0.174***\n(0.018)\n-0.174***\n(0.015)\n\n\nfe\n\n\nf2\n-\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n997\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.489\n0.659\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe sw0() and csw0() syntax are similar to sw() and csw(), but start with a model that excludes the variables specified in sw() and csw():\n\nfit = pf.feols(\"Y ~ X1 + X2 | sw0(f1, f2)\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1), fit.fetch_model(2)])\n\nModel: Y~X1+X2\nModel: Y~X1+X2|f1\nModel: Y~X1+X2|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\nX1\n-0.993***\n(0.082)\n-0.950***\n(0.067)\n-0.979***\n(0.077)\n\n\nX2\n-0.176***\n(0.022)\n-0.174***\n(0.018)\n-0.175***\n(0.022)\n\n\nIntercept\n0.889***\n(0.108)\n\n\n\n\nfe\n\n\nf2\n-\n-\nx\n\n\nf1\n-\nx\n-\n\n\nstats\n\n\nObservations\n998\n997\n998\n\n\nS.E. type\niid\nby: f1\nby: f2\n\n\nR2\n0.177\n0.489\n0.354\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nThe feols() function also supports multiple dependent variables. The following example estimates two models, one with Y1 as the dependent variable and one with Y2 as the dependent variable.\n\nfit = pf.feols(\"Y + Y2 ~ X1 | f1 + f2\", data)\npf.etable([fit.fetch_model(0), fit.fetch_model(1)])\n\nModel: Y~X1|f1+f2\nModel: Y2~X1|f1+f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\n\n\n(1)\n(2)\n\n\n\n\ncoef\n\n\nX1\n-0.919***\n(0.065)\n-1.228***\n(0.195)\n\n\nfe\n\n\nf2\nx\nx\n\n\nf1\nx\nx\n\n\nstats\n\n\nObservations\n997\n998\n\n\nS.E. type\nby: f1\nby: f1\n\n\nR2\n0.609\n0.168\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIt is possible to combine different multiple estimation operators:\n\nfit = pf.feols(\"Y + Y2 ~ X1 | sw(f1, f2)\", data)\npf.etable([fit.fetch_model(0),\n fit.fetch_model(1),\n fit.fetch_model(2),\n fit.fetch_model(3)\n ]\n )\n\nModel: Y~X1|f1\nModel: Y2~X1|f1\nModel: Y~X1|f2\nModel: Y2~X1|f2\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nY\nY2\nY\nY2\n\n\n(1)\n(2)\n(3)\n(4)\n\n\n\n\ncoef\n\n\nX1\n-0.949***\n(0.069)\n-1.266***\n(0.176)\n-0.982***\n(0.081)\n-1.301***\n(0.205)\n\n\nfe\n\n\nf2\n-\n-\nx\nx\n\n\nf1\nx\nx\n-\n-\n\n\nstats\n\n\nObservations\n997\n998\n998\n999\n\n\nS.E. type\nby: f1\nby: f1\nby: f2\nby: f2\n\n\nR2\n0.437\n0.115\n0.302\n0.090\n\n\n\n\n\n\n\n\n\n\n\n \n\n\nIn general, using muliple estimation syntax can improve the estimation time as covariates that are demeaned in one model and are used in another model do not need to be demeaned again: feols() implements a caching mechanism that stores the demeaned covariates.\nBesides OLS, feols() also supports IV estimation via three part formulas:\n\nfit = pf.feols(\"Y ~ X2 | f1 + f2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nX1\n-1.050097\n0.085493\n-12.282912\n5.133671e-13\n-1.224949\n-0.875245\n\n\nX2\n-0.174351\n0.014779\n-11.797039\n1.369793e-12\n-0.204578\n-0.144124\n\n\n\n\n\n\n\nHere, X1 is the endogenous variable and Z1 is the instrument. f1 and f2 are the fixed effects, as before. To estimate IV models without fixed effects, simply omit the fixed effects part of the formula:\n\nfit = pf.feols(\"Y ~ X2 | X1 ~ Z1\", data)\nfit.tidy()\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\nCoefficient\n\n\n\n\n\n\n\n\n\n\nIntercept\n0.861939\n0.151187\n5.701137\n1.567858e-08\n0.565257\n1.158622\n\n\nX1\n-0.967238\n0.130078\n-7.435847\n2.238210e-13\n-1.222497\n-0.711980\n\n\nX2\n-0.176416\n0.021769\n-8.104001\n1.554312e-15\n-0.219134\n-0.133697\n\n\n\n\n\n\n\nLast, feols() supports interaction of variables via the i() syntax. Documentation on this is tba.\nAfter fitting a model via feols(), you can use the predict() method to get the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict()[0:5]\n\narray([ 3.0633663 , -0.69574133, -0.91240433, -0.46370257, -1.67331154])\n\n\nThe predict() method also supports a newdata argument to predict on new data, which returns a numpy array of the predicted values:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.predict(newdata=data)[0:5]\n\narray([ 2.14598761, nan, nan, 3.06336415, -0.69574276])\n\n\nLast, you can plot the results of a model via the coefplot() method:\n\nfit = pf.feols(\"Y ~ X1 + X2 | f1 + f2\", data)\nfit.coefplot()\n\n \n \n\n\nObjects of type Feols support a range of other methods to conduct inference. For example, you can run a wild (cluster) bootstrap via the wildboottest() method:\n\nfit.wildboottest(param = \"X1\", reps=1000)\n\nparam X1\nt value -14.70814685400939\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(f1)\nimpose_null True\ndtype: object\n\n\nwould run a wild bootstrap test for the coefficient of X1 with 1000 bootstrap repetitions.\nFor a wild cluster bootstrap, you can specify the cluster variable via the cluster argument:\n\nfit.wildboottest(param = \"X1\", reps=1000, cluster=\"group_id\")\n\nparam X1\nt value -13.658130940490494\nPr(>|t|) 0.0\nbootstrap_type 11\ninference CRV(group_id)\nimpose_null True\ndtype: object\n\n\nThe ritest() method can be used to conduct randomization inference:\n\nfit.ritest(resampvar = \"X1\", reps=1000)\n\nH0 X1=0\nri-type randomization-c\nEstimate -0.9240461507764967\nPr(>|t|) 0.0\nStd. Error (Pr(>|t|)) 0.0\n2.5% (Pr(>|t|)) 0.0\n97.5% (Pr(>|t|)) 0.0\ndtype: object\n\n\nLast, you can compute the cluster causal variance estimator by Athey et al by using the ccv() method:\n\nimport numpy as np\nrng = np.random.default_rng(1234)\ndata[\"D\"] = rng.choice([0, 1], size = data.shape[0])\nfit_D = pf.feols(\"Y ~ D\", data = data)\nfit_D.ccv(treatment = \"D\", cluster = \"group_id\")\n\n/home/runner/work/pyfixest/pyfixest/pyfixest/estimation/feols_.py:1365: UserWarning: The initial model was not clustered. CRV1 inference is computed and stored in the model object.\n warnings.warn(\n\n\n\n\n\n\n\n\n\nEstimate\nStd. Error\nt value\nPr(>|t|)\n2.5%\n97.5%\n\n\n\n\nCCV\n0.016087657906364183\n0.25692\n0.062617\n0.950761\n-0.523682\n0.555857\n\n\nCRV1\n0.016088\n0.13378\n0.120254\n0.905614\n-0.264974\n0.29715",
"crumbs": [
"Function Reference",
"Estimation Functions",
@@ -1435,14 +1435,14 @@
"href": "table-layout.html#basic-usage",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage",
- "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
+ "text": "Basic Usage\nWe can compare all regression models via the pyfixest-internal pf.etable() function:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:\n\npf.etable(pf.feols(\"Y+Y2~csw(X1,X2,X1:X2)\", data=data))\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -1.000*** (0.085)\n -0.993*** (0.082)\n -0.992*** (0.082)\n -1.322*** (0.215)\n -1.316*** (0.214)\n -1.316*** (0.215)\n \n \n X2\n \n -0.176*** (0.022)\n -0.197*** (0.036)\n \n -0.133* (0.057)\n -0.132 (0.095)\n \n \n X1:X2\n \n \n 0.020 (0.027)\n \n \n -0.001 (0.071)\n \n \n Intercept\n 0.919*** (0.112)\n 0.889*** (0.108)\n 0.888*** (0.108)\n 1.064*** (0.283)\n 1.042*** (0.283)\n 1.042*** (0.283)\n \n \n stats\n \n \n Observations\n 998\n 998\n 998\n 999\n 999\n 999\n \n \n S.E. type\n iid\n iid\n iid\n iid\n iid\n iid\n \n \n R2\n 0.123\n 0.177\n 0.177\n 0.037\n 0.042\n 0.042"
},
{
"objectID": "table-layout.html#keep-and-drop-variables",
"href": "table-layout.html#keep-and-drop-variables",
"title": "Regression Tables via pf.etable()",
"section": "Keep and drop variables",
- "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Keep and drop variables\netable allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=\"X1\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can use the exact_match argument to select a specific set of variables:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=[\"X1\", \"X2\"], exact_match=True)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nWe can also easily drop variables via the drop argument:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=[\"X1\"])\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#hide-fixed-effects-or-se-type-rows",
@@ -1456,49 +1456,49 @@
"href": "table-layout.html#display-p-values-or-confidence-intervals",
"title": "Regression Tables via pf.etable()",
"section": "Display p-values or confidence intervals",
- "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Display p-values or confidence intervals\nBy default, pf.etable() reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt function argument.\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt=\"b \\n (se) \\n [p]\")\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067) [0.000]\n -0.924*** (0.061) [0.000]\n -0.924*** (0.061) [0.000]\n -1.267*** (0.174) [0.000]\n -1.232*** (0.192) [0.000]\n -1.231*** (0.192) [0.000]\n \n \n X2\n -0.174*** (0.018) [0.000]\n -0.174*** (0.015) [0.000]\n -0.185*** (0.025) [0.000]\n -0.131** (0.042) [0.005]\n -0.118** (0.042) [0.008]\n -0.074 (0.104) [0.482]\n \n \n X1:X2\n \n \n 0.011 (0.018) [0.565]\n \n \n -0.041 (0.081) [0.618]\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#significance-levels-and-rounding",
"href": "table-layout.html#significance-levels-and-rounding",
"title": "Regression Tables via pf.etable()",
"section": "Significance levels and rounding",
- "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
+ "text": "Significance levels and rounding\nAdditionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code and digits function arguments:\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180"
},
{
"objectID": "table-layout.html#other-output-formats",
"href": "table-layout.html#other-output-formats",
"title": "Regression Tables via pf.etable()",
"section": "Other output formats",
- "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf2 - x x - x x\nf1 x x x x x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
+ "text": "Other output formats\nBy default, pf.etable() returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type argument.\n\n# Pandas styler output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n coef_fmt=\"b (se)\",\n type=\"df\",\n)\n\n\n\n\n \n \n \n est1\n est2\n est3\n est4\n est5\n est6\n \n \n \n \n depvar\n Y\n Y\n Y\n Y2\n Y2\n Y2\n \n \n X1\n -0.94953*** (0.06652)\n -0.92405*** (0.06093)\n -0.92417*** (0.06094)\n -1.26655*** (0.17359)\n -1.23153*** (0.19228)\n -1.23100*** (0.19167)\n \n \n X2\n -0.17423*** (0.01840)\n -0.17411*** (0.01461)\n -0.18550*** (0.02516)\n -0.13056*** (0.04239)\n -0.11767*** (0.04152)\n -0.07369 (0.10356)\n \n \n X1:X2\n \n \n 0.01057 (0.01818)\n \n \n -0.04082 (0.08093)\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.48899\n 0.65904\n 0.65916\n 0.12017\n 0.17151\n 0.17180\n \n \n\n\n\n\n\n# Markdown output:\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=5,\n type=\"md\",\n)\n\nindex est1 est2 est3 est4 est5 est6\n------------ ------------ ------------ ------------ ------------ ------------ ------------\ndepvar Y Y Y Y2 Y2 Y2\n------------------------------------------------------------------------------------------------\nX1 -0.94953*** -0.92405*** -0.92417*** -1.26655*** -1.23153*** -1.23100***\n (0.06652) (0.06093) (0.06094) (0.17359) (0.19228) (0.19167)\nX2 -0.17423*** -0.17411*** -0.18550*** -0.13056*** -0.11767*** -0.07369\n (0.01840) (0.01461) (0.02516) (0.04239) (0.04152) (0.10356)\nX1:X2 0.01057 -0.04082\n (0.01818) (0.08093)\n------------------------------------------------------------------------------------------------\nf1 x x x x x x\nf2 - x x - x x\n------------------------------------------------------------------------------------------------\nObservations 997 997 997 998 998 998\nS.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1\nR2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180\n------------------------------------------------------------------------------------------------\n\n\n\nTo obtain latex output use format = \"tex\". If you want to save the table as a tex file, you can use the filename= argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True argument. Etable will use latex packages booktabs, threeparttable and makecell for the table layout, so don’t forget to include these packages in your latex document.\n\n# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):\ntab = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n signif_code=[0.01, 0.05, 0.1],\n digits=2,\n type=\"tex\",\n print_tex=True,\n)\n\nThe following code generates a pdf including the regression table which you can display clicking on the link below the cell:\n\n## Use pylatex to create a tex file with the table\n\n\ndef make_pdf(tab, file):\n \"Create a PDF document with tex table.\"\n doc = pl.Document()\n doc.packages.append(pl.Package(\"booktabs\"))\n doc.packages.append(pl.Package(\"threeparttable\"))\n doc.packages.append(pl.Package(\"makecell\"))\n\n with (\n doc.create(pl.Section(\"A PyFixest LateX Table\")),\n doc.create(pl.Table(position=\"htbp\")) as table,\n ):\n table.append(pl.NoEscape(tab))\n\n doc.generate_pdf(file, clean_tex=False)\n\n\n# Compile latex to pdf & display a button with the hyperlink to the pdf\n# requires tex installation\nrun = False\nif run:\n make_pdf(tab, \"latexdocs/SampleTableDoc\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc.pdf\"))\n\nlatexdocs/SampleTableDoc.pdf"
},
{
"objectID": "table-layout.html#rename-variables",
"href": "table-layout.html#rename-variables",
"title": "Regression Tables via pf.etable()",
"section": "Rename variables",
- "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Rename variables\nYou can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).\n\nlabels = {\n \"Y\": \"Wage\",\n \"Y2\": \"Wealth\",\n \"X1\": \"Age\",\n \"X2\": \"Years of Schooling\",\n \"f1\": \"Industry\",\n \"f2\": \"Year\",\n}\n\npf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nIf you want to label the rows indicating the inclusion of fixed effects not with the variable label but with a custom label, you can pass on a separate dictionary to the felabels argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-model-headlines",
"href": "table-layout.html#custom-model-headlines",
"title": "Regression Tables via pf.etable()",
"section": "Custom model headlines",
- "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
+ "text": "Custom model headlines\nYou can also add custom headers for each model by passing a list of strings to the model_headers argument.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n model_heads=[\"US\", \"China\", \"EU\", \"US\", \"China\", \"EU\"],\n)\n\n\n\n\n\n\n\n \n \n \n \n Wage\n \n \n Wealth\n \n\n\n \n \n US\n \n \n China\n \n \n EU\n \n \n US\n \n \n China\n \n \n EU\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nOr change the ordering of headlines having headlines first and then dependent variables using the head_order argument. “hd” stands for headlines then dependent variables, “dh” for dependent variables then headlines. Assigning “d” or “h” can be used to only show dependent variables or only headlines. When head_order=“” only model numbers are shown.\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nRemove the dependent variables from the headers:\n\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"\",\n)\n\n\n\n\n\n\n\n \n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172"
},
{
"objectID": "table-layout.html#further-custom-model-information",
"href": "table-layout.html#further-custom-model-information",
"title": "Regression Tables via pf.etable()",
"section": "Further custom model information",
- "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Further custom model information\nYou can add further custom model statistics/information to the bottom of the table by using the custom_stats argument to which you pass a dictionary with the name of the row and lists of values. The length of the lists must be equal to the number of models.\n\npf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n custom_model_stats={\n \"Number of Clusters\": [42, 42, 42, 37, 37, 37],\n \"Additional Info\": [\"A\", \"A\", \"B\", \"B\", \"C\", \"C\"],\n },\n)\n\n\n\n\n\n\n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Number of Clusters\n 42\n 42\n 42\n 37\n 37\n 37\n \n \n Additional Info\n A\n A\n B\n B\n C\n C\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#custom-table-notes",
"href": "table-layout.html#custom-table-notes",
"title": "Regression Tables via pf.etable()",
"section": "Custom table notes",
- "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
+ "text": "Custom table notes\nYou can replace the default table notes with your own notes using the notes argument.\n\nmynotes = \"Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.\"\npf.etable(\n [fit1, fit4, fit2, fit5, fit3, fit6],\n labels=labels,\n model_heads=[\"US\", \"US\", \"China\", \"China\", \"EU\", \"EU\"],\n head_order=\"hd\",\n notes=mynotes,\n)\n\n\n\n\n\n\n\n \n \n \n \n US\n \n \n China\n \n \n EU\n \n\n\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -1.267*** (0.174)\n -0.924*** (0.061)\n -1.232*** (0.192)\n -0.924*** (0.061)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.131** (0.042)\n -0.174*** (0.015)\n -0.118** (0.042)\n -0.185*** (0.025)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n \n \n 0.011 (0.018)\n -0.041 (0.081)\n \n \n fe\n \n \n Industry\n x\n x\n x\n x\n x\n x\n \n \n Year\n -\n -\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 998\n 997\n 998\n 997\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.120\n 0.659\n 0.172\n 0.659\n 0.172\n \n\n \n \n \n Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet."
},
{
"objectID": "table-layout.html#publication-ready-latex-tables",
@@ -1519,49 +1519,49 @@
"href": "table-layout.html#summarize-by-characteristics-in-columns-and-rows",
"title": "Regression Tables via pf.etable()",
"section": "Summarize by characteristics in columns and rows",
- "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.19\n 2.35\n -0.04\n 2.21\n -0.05\n 2.28\n -0.23\n 2.39\n \n \n Wealth\n -0.72\n 5.71\n 0.05\n 4.99\n -0.09\n 5.53\n -0.56\n 6.06\n \n \n Age\n 1.07\n 0.81\n 0.96\n 0.80\n 1.08\n 0.81\n 1.06\n 0.81\n \n \n Years of Schooling\n 0.05\n 3.10\n -0.28\n 2.86\n -0.18\n 3.13\n -0.09\n 3.08\n \n \n nobs\n \n \n Number of observations\n 229\n \n 244\n \n 270\n \n 254\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.19(2.35)\n -0.04(2.21)\n -0.05(2.28)\n -0.23(2.39)\n \n \n Wealth\n -0.72(5.71)\n 0.05(4.99)\n -0.09(5.53)\n -0.56(6.06)\n \n \n Age\n 1.07(0.81)\n 0.96(0.80)\n 1.08(0.81)\n 1.06(0.81)\n \n \n Years of Schooling\n 0.05(3.10)\n -0.28(2.86)\n -0.18(3.13)\n -0.09(3.08)\n \n \n nobs\n \n \n Number of observations\n 229\n 244\n 270\n 254\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 229\n -0.19\n 2.35\n 270\n -0.05\n 2.28\n \n \n Wealth\n 229\n -0.72\n 5.71\n 270\n -0.09\n 5.53\n \n \n Age\n 229\n 1.07\n 0.81\n 270\n 1.08\n 0.81\n \n \n Years of Schooling\n 229\n 0.05\n 3.10\n 270\n -0.18\n 3.13\n \n \n White collar\n \n \n Wage\n 244\n -0.04\n 2.21\n 254\n -0.23\n 2.39\n \n \n Wealth\n 244\n 0.05\n 4.99\n 254\n -0.56\n 6.06\n \n \n Age\n 244\n 0.96\n 0.80\n 254\n 1.06\n 0.81\n \n \n Years of Schooling\n 244\n -0.28\n 2.86\n 254\n -0.09\n 3.08\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
+ "text": "Summarize by characteristics in columns and rows\nYou can summarize by characteristics using the bycol argument when groups are to be displayed in columns. When the number of observations is the same for all variables in a group, you can also opt to display the number of observations only once for each group byin a separate line at the bottom of the table with counts_row_below==True.\n\n# Generate some categorial data\ndata[\"country\"] = np.random.choice([\"US\", \"EU\"], data.shape[0])\ndata[\"occupation\"] = np.random.choice([\"Blue collar\", \"White collar\"], data.shape[0])\n\n# Drop nan values to have balanced data\ndata.dropna(inplace=True)\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n \n \n EU\n \n \n US\n \n\n\n \n \n Blue collar\n \n \n White collar\n \n \n Blue collar\n \n \n White collar\n \n\n\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n Mean\n Std. Dev.\n\n\n\n \n stats\n \n \n Wage\n -0.18\n 2.35\n 0.10\n 2.26\n -0.28\n 2.33\n -0.13\n 2.28\n \n \n Wealth\n -0.33\n 5.51\n -0.10\n 5.54\n -0.39\n 5.48\n -0.45\n 5.83\n \n \n Age\n 1.05\n 0.82\n 1.05\n 0.81\n 1.02\n 0.81\n 1.05\n 0.80\n \n \n Years of Schooling\n -0.14\n 2.99\n -0.31\n 2.92\n 0.12\n 3.24\n -0.19\n 3.02\n \n \n nobs\n \n \n Number of observations\n 254\n \n 242\n \n 246\n \n 255\n \n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nYou can also use custom aggregation functions to compute further statistics or affect how statistics are presented. Pyfixest provides two such functions mean_std and mean_newline_std which compute the mean and standard deviation and display both the same cell (either with line break between them or not). This allows to have more compact tables when you want to show statistics for many characteristcs in the columns.\nYou can also hide the display of the statistics labels in the header with hide_stats_labels=True. In that case a table note will be added naming the statistics displayed using its label (if you have not provided a custom note).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\", \"occupation\"],\n stats=[\"mean_newline_std\", \"count\"],\n caption=\"Descriptive statistics\",\n stats_labels={\"count\": \"Number of observations\"},\n counts_row_below=True,\n hide_stats=True,\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n Blue collar\n White collar\n Blue collar\n White collar\n\n\n\n \n stats\n \n \n Wage\n -0.18(2.35)\n 0.10(2.26)\n -0.28(2.33)\n -0.13(2.28)\n \n \n Wealth\n -0.33(5.51)\n -0.10(5.54)\n -0.39(5.48)\n -0.45(5.83)\n \n \n Age\n 1.05(0.82)\n 1.05(0.81)\n 1.02(0.81)\n 1.05(0.80)\n \n \n Years of Schooling\n -0.14(2.99)\n -0.31(2.92)\n 0.12(3.24)\n -0.19(3.02)\n \n \n nobs\n \n \n Number of observations\n 254\n 242\n 246\n 255\n \n\n \n \n \n Note: Displayed statistics are Mean (Std. Dev.).\n \n\n\n\n\n\n\n \n\n\nYou can also split by characteristics in both columns and rows. Note that you can only use one grouping variable in rows, but several in columns (as shown above).\n\npf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n caption=\"Descriptive statistics\",\n)\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n \n EU\n \n \n US\n \n\n\n N\n Mean\n Std. Dev.\n N\n Mean\n Std. Dev.\n\n\n\n \n Blue collar\n \n \n Wage\n 254\n -0.18\n 2.35\n 246\n -0.28\n 2.33\n \n \n Wealth\n 254\n -0.33\n 5.51\n 246\n -0.39\n 5.48\n \n \n Age\n 254\n 1.05\n 0.82\n 246\n 1.02\n 0.81\n \n \n Years of Schooling\n 254\n -0.14\n 2.99\n 246\n 0.12\n 3.24\n \n \n White collar\n \n \n Wage\n 242\n 0.10\n 2.26\n 255\n -0.13\n 2.28\n \n \n Wealth\n 242\n -0.10\n 5.54\n 255\n -0.45\n 5.83\n \n \n Age\n 242\n 1.05\n 0.81\n 255\n 1.05\n 0.80\n \n \n Years of Schooling\n 242\n -0.31\n 2.92\n 255\n -0.19\n 3.02\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\nAnd you can again export descriptive statistics tables also to LaTex:\n\ndtab = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n labels=labels,\n bycol=[\"country\"],\n byrow=\"occupation\",\n stats=[\"count\", \"mean\", \"std\"],\n type=\"tex\",\n)\n\nrun = False\nif run:\n make_pdf(dtab, \"latexdocs/SampleTableDoc3\")\ndisplay(FileLink(\"latexdocs/SampleTableDoc3.pdf\"))\n\nlatexdocs/SampleTableDoc3.pdf"
},
{
"objectID": "table-layout.html#basic-usage-of-make_table",
"href": "table-layout.html#basic-usage-of-make_table",
"title": "Regression Tables via pf.etable()",
"section": "Basic Usage of make_table",
- "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n -0.04\n -1.77\n -1.68\n 1.89\n \n \n 1\n 0.39\n 1.05\n -0.07\n -1.59\n \n \n 2\n 1.1\n -0.18\n 0.3\n -1.25\n \n \n 3\n 1.78\n -0.18\n -0.09\n -0.08\n \n\n \n \n \n These are notes"
+ "text": "Basic Usage of make_table\n\ndf = pd.DataFrame(np.random.randn(4, 4).round(2), columns=[\"A\", \"B\", \"C\", \"D\"])\n\n# Make Booktabs style table\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n A\n B\n C\n D\n\n\n\n \n 0\n 0.87\n -1.25\n -1.78\n 1.12\n \n \n 1\n -0.88\n 1.08\n -0.47\n -0.5\n \n \n 2\n -0.31\n 1.04\n 0.56\n 0.12\n \n \n 3\n 0.37\n -0.63\n -0.72\n 1.4\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#mutiindex-dataframes",
"href": "table-layout.html#mutiindex-dataframes",
"title": "Regression Tables via pf.etable()",
"section": "Mutiindex DataFrames",
- "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -1.275\n -0.179\n -0.075\n -1.221\n -0.719\n -0.98\n -0.073\n -2.271\n \n \n Variable 2\n 0.052\n 1.195\n -0.214\n 0.507\n -0.592\n 0.87\n -0.654\n -1.502\n \n \n Variable 3\n 0.379\n -0.647\n -0.001\n 1.133\n 0.793\n -0.833\n -1.638\n 1.531\n \n \n Group 2\n \n \n Variable 4\n -0.618\n 1.28\n -0.591\n 0.24\n -1.099\n -0.131\n 0.299\n 2.096\n \n \n Variable 5\n 0.087\n 0.534\n 0.158\n -0.036\n -0.609\n 0.494\n 1.386\n 1.375\n \n \n Group 3\n \n \n Variable 6\n -1.231\n 0.547\n 0.478\n -1.153\n -0.298\n -1.565\n -0.048\n -0.198\n \n\n \n \n \n These are notes"
+ "text": "Mutiindex DataFrames\nWhen the respective dataframe has a mutiindex for the columns, columns spanners are generated from the index. The row index can also be a multiindex (of at most two levels). In this case the first index level is used to generate group rows (for instance using the index name as headline and separating the groups by a horizontal line) and the second index level is used to generate the row labels.\n\n# Create a multiindex dataframe with random data\nrow_index = pd.MultiIndex.from_tuples(\n [\n (\"Group 1\", \"Variable 1\"),\n (\"Group 1\", \"Variable 2\"),\n (\"Group 1\", \"Variable 3\"),\n (\"Group 2\", \"Variable 4\"),\n (\"Group 2\", \"Variable 5\"),\n (\"Group 3\", \"Variable 6\"),\n ]\n)\n\ncol_index = pd.MultiIndex.from_product([[\"A\", \"B\"], [\"X\", \"Y\"], [\"High\", \"Low\"]])\ndf = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)\n\npf.make_table(df=df, caption=\"This is a caption\", notes=\"These are notes\")\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes\n \n\n\n\n\n\n\n \n\n\nYou can also hide column group names: This just creates a table where variables on the second level of the row index are displayed in groups based on the first level separated by horizontal lines.\n\npf.make_table(\n df=df, caption=\"This is a caption\", notes=\"These are notes\", rgroup_display=False\n).tab_style(style=style.text(style=\"italic\"), locations=loc.body(rows=[1, 5]))\n\n\n\n\n\n\n \n This is a caption\n \n\n \n \n \n \n A\n \n \n B\n \n\n\n \n \n X\n \n \n Y\n \n \n X\n \n \n Y\n \n\n\n High\n Low\n High\n Low\n High\n Low\n High\n Low\n\n\n\n \n Group 1\n \n \n Variable 1\n -0.62\n 0.188\n 1.106\n -1.714\n -0.211\n -0.724\n -1.071\n -1.315\n \n \n Variable 2\n -1.684\n -0.286\n 0.953\n 1.751\n -0.988\n 0.258\n 0.584\n 1.655\n \n \n Variable 3\n 1.445\n -0.804\n 0.054\n -1.11\n 0.045\n -0.204\n -0.703\n -0.557\n \n \n Group 2\n \n \n Variable 4\n 0.982\n -0.25\n -1.133\n 1.313\n 0.426\n 0.521\n -0.363\n -0.054\n \n \n Variable 5\n 0.637\n -1.569\n -1.251\n -0.262\n 1.247\n 0.335\n -0.588\n -0.68\n \n \n Group 3\n \n \n Variable 6\n -1.39\n -1.201\n -2.031\n -0.74\n -0.839\n 1.171\n -0.278\n 0.756\n \n\n \n \n \n These are notes"
},
{
"objectID": "table-layout.html#example-styling",
"href": "table-layout.html#example-styling",
"title": "Regression Tables via pf.etable()",
"section": "Example Styling",
- "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Example Styling\n\n(\n pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])\n .tab_options(\n column_labels_background_color=\"cornsilk\",\n stub_background_color=\"whitesmoke\",\n )\n .tab_style(\n style=style.fill(color=\"mistyrose\"),\n locations=loc.body(columns=\"(3)\", rows=[\"X2\"]),\n )\n)\n\n\n\n\n\n\n\n \n \n Y\n \n \n Y2\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n X1\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n X2\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n X1:X2\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n f1\n x\n x\n x\n x\n x\n x\n \n \n f2\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "table-layout.html#defining-table-styles-some-examples",
"href": "table-layout.html#defining-table-styles-some-examples",
"title": "Regression Tables via pf.etable()",
"section": "Defining Table Styles: Some Examples",
- "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
+ "text": "Defining Table Styles: Some Examples\nYou can easily define table styles that you can apply to all tables in your project. Just define a dictionary with the respective values for the tab options (see the Great Tables documentation) and use the style with .tab_options(**style_dict).\n\nstyle_print = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_body_border_bottom_width\": \"1px\",\n \"column_labels_border_top_width\": \"1px\",\n \"table_width\": \"14cm\",\n}\n\n\nstyle_presentation = {\n \"table_font_size\": \"16px\",\n \"table_font_color_light\": \"white\",\n \"table_body_border_top_style\": \"hidden\",\n \"table_body_border_bottom_style\": \"hidden\",\n \"heading_title_font_size\": \"18px\",\n \"source_notes_font_size\": \"12px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"6px\",\n \"column_labels_background_color\": \"midnightblue\",\n \"stub_background_color\": \"whitesmoke\",\n \"row_group_background_color\": \"whitesmoke\",\n \"table_background_color\": \"whitesmoke\",\n \"heading_background_color\": \"white\",\n \"source_notes_background_color\": \"white\",\n \"column_labels_border_bottom_color\": \"white\",\n \"column_labels_font_weight\": \"bold\",\n \"row_group_font_weight\": \"bold\",\n \"table_width\": \"18cm\",\n}\n\n\nt1 = pf.dtable(\n data,\n vars=[\"Y\", \"Y2\", \"X1\", \"X2\"],\n stats=[\"count\", \"mean\", \"std\", \"min\", \"max\"],\n labels=labels,\n caption=\"Descriptive statistics\",\n)\n\nt2 = pf.etable(\n [fit1, fit2, fit3, fit4, fit5, fit6],\n labels=labels,\n show_se=False,\n felabels={\"f1\": \"Industry Fixed Effects\", \"f2\": \"Year Fixed Effects\"},\n caption=\"Regression results\",\n)\n\n\ndisplay(t1.tab_options(**style_print))\ndisplay(t2.tab_options(**style_print))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\nstyle_printDouble = {\n \"table_font_size\": \"12px\",\n \"heading_title_font_size\": \"12px\",\n \"source_notes_font_size\": \"8px\",\n \"data_row_padding\": \"3px\",\n \"column_labels_padding\": \"3px\",\n \"table_body_border_bottom_style\": \"double\",\n \"column_labels_border_top_style\": \"double\",\n \"column_labels_border_bottom_width\": \"0.5px\",\n \"row_group_border_top_style\": \"hidden\",\n \"table_body_border_top_style\": \"None\",\n \"table_width\": \"14cm\",\n}\ndisplay(t1.tab_options(**style_printDouble))\ndisplay(t2.tab_options(**style_printDouble))\n\n\n\n\n\n\n \n Descriptive statistics\n \n\n \n N\n Mean\n Std. Dev.\n Min\n Max\n\n\n\n \n Wage\n 997\n -0.13\n 2.31\n -6.54\n 6.91\n \n \n Wealth\n 997\n -0.32\n 5.59\n -16.97\n 17.16\n \n \n Age\n 997\n 1.04\n 0.81\n 0.00\n 2.00\n \n \n Years of Schooling\n 997\n -0.13\n 3.05\n -9.67\n 10.99\n \n\n \n \n \n \n \n\n\n\n\n\n\n \n\n\n\n\n\n\n\n \n Regression results\n \n\n \n \n Wage\n \n \n Wealth\n \n\n\n (1)\n (2)\n (3)\n (4)\n (5)\n (6)\n\n\n\n \n coef\n \n \n Age\n -0.950*** (0.067)\n -0.924*** (0.061)\n -0.924*** (0.061)\n -1.267*** (0.174)\n -1.232*** (0.192)\n -1.231*** (0.192)\n \n \n Years of Schooling\n -0.174*** (0.018)\n -0.174*** (0.015)\n -0.185*** (0.025)\n -0.131** (0.042)\n -0.118** (0.042)\n -0.074 (0.104)\n \n \n Age × Years of Schooling\n \n \n 0.011 (0.018)\n \n \n -0.041 (0.081)\n \n \n fe\n \n \n Industry Fixed Effects\n x\n x\n x\n x\n x\n x\n \n \n Year Fixed Effects\n -\n x\n x\n -\n x\n x\n \n \n stats\n \n \n Observations\n 997\n 997\n 997\n 998\n 998\n 998\n \n \n S.E. type\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n by: f1\n \n \n R2\n 0.489\n 0.659\n 0.659\n 0.120\n 0.172\n 0.172"
},
{
"objectID": "replicating-the-effect.html",
"href": "replicating-the-effect.html",
"title": "Replicating Examples from “The Effect”",
"section": "",
- "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\ncausaldata: 0.1.4\npyfixest : 0.24.3"
+ "text": "This notebook replicates code examples from Nick Huntington-Klein’s book on causal inference, The Effect.\nfrom causaldata import Mroz, gapminder, organ_donations, restaurant_inspections\n\nimport pyfixest as pf\n\n%load_ext watermark\n%watermark --iversions\n\n\n \n \n \n\n\n\n \n \n \n\n\npyfixest : 0.24.3\ncausaldata: 0.1.4"
},
{
"objectID": "replicating-the-effect.html#chapter-4-describing-relationships",
"href": "replicating-the-effect.html#chapter-4-describing-relationships",
"title": "Replicating Examples from “The Effect”",
"section": "Chapter 4: Describing Relationships",
- "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4568/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
+ "text": "Chapter 4: Describing Relationships\n\n# Read in data\ndt = Mroz.load_pandas().data\n# Keep just working women\ndt = dt.query(\"lfp\")\n# Create unlogged earnings\ndt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n# 5. Run multiple linear regression models by succesively adding controls\nfit = pf.feols(fml=\"lwg ~ csw(inc, wc, k5)\", data=dt, vcov=\"iid\")\npf.etable(fit)\n\n/tmp/ipykernel_4654/786816010.py:6: SettingWithCopyWarning: \nA value is trying to be set on a copy of a slice from a DataFrame.\nTry using .loc[row_indexer,col_indexer] = value instead\n\nSee the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n dt.loc[:, \"earn\"] = dt[\"lwg\"].apply(\"exp\")\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\nlwg\n\n\n(1)\n(2)\n(3)\n\n\n\n\ncoef\n\n\ninc\n0.010**\n(0.003)\n0.005\n(0.003)\n0.005\n(0.003)\n\n\nwc\n\n0.342***\n(0.075)\n0.349***\n(0.075)\n\n\nk5\n\n\n-0.072\n(0.087)\n\n\nIntercept\n1.007***\n(0.071)\n0.972***\n(0.070)\n0.982***\n(0.071)\n\n\nstats\n\n\nObservations\n428\n428\n428\n\n\nS.E. type\niid\niid\niid\n\n\nR2\n0.020\n0.066\n0.068"
},
{
"objectID": "replicating-the-effect.html#chapter-13-regression",
@@ -1624,7 +1624,7 @@
"href": "difference-in-differences.html#inspecting-the-outcome-variable",
"title": "Difference-in-Differences Estimation",
"section": "Inspecting the Outcome Variable",
- "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n figsize=(2.5, 0.8),\n title = \"Outcome Plot\"\n)"
+ "text": "Inspecting the Outcome Variable\npf.panelview() further allows us to inspect the “outcome” variable over time:\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n collapse_to_cohort=True,\n title = \"Outcome Plot\"\n)\n\n\n\n\n\n\n\n\nWe immediately see that the first cohort is switched into treatment in 2000, while the second cohort is switched into treatment by 2010. Before each cohort is switched into treatment, the trends are parallel.\nWe can additionally inspect individual units by dropping the collapse_to_cohort argument. Because we have a large sample, we might want to inspect only a subset of units.\n\npf.panelview(\n df_multi_cohort,\n outcome=\"dep_var\",\n unit=\"unit\",\n time=\"year\",\n treat=\"treat\",\n subsamp=100,\n title = \"Outcome Plot\"\n)"
},
{
"objectID": "difference-in-differences.html#one-shot-adoption-static-and-dynamic-specifications",
diff --git a/table-layout.html b/table-layout.html
index 3dd0c424..2773792b 100644
--- a/table-layout.html
+++ b/table-layout.html
@@ -245,7 +245,7 @@ Regression Tables via pf.etable()
Table Layout with PyFixest
Pyfixest comes with functions to generate publication-ready tables. Regression tables are generated with pf.etable()
, which can output different formats, for instance using the Great Tables package or generating formatted LaTex Tables using booktabs. There are also further functions pf.dtable()
to display descriptive statistics and pf.make_table()
generating formatted tables from pandas dataframes in the same layout.
To begin, we load some libraries and fit a set of regression models.
-
+
import numpy as np
import pandas as pd
import pylatex as pl # for the latex table; note: not a dependency of pyfixest - needs manual installation
@@ -267,7 +267,7 @@ Table Layout wi
= pf.feols("Y2 ~ X1 *X2 | f1 + f2", data=data) fit6
-
+
@@ -301,7 +301,7 @@ Table Layout wi
-
+
@@ -338,54 +338,54 @@ Table Layout wi
Basic Usage
We can compare all regression models via the pyfixest-internal pf.etable()
function:
-
+
pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
-
+
@@ -444,20 +444,20 @@ Basic Usage
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -507,54 +507,54 @@ Basic Usage
You can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:
-
+
"Y+Y2~csw(X1,X2,X1:X2)", data=data)) pf.etable(pf.feols(
-
+
@@ -667,54 +667,54 @@ Basic Usage
Keep and drop variables
etable
allows us to do a few things out of the box. For example, we can only keep the variables that we’d like, which keeps all variables that fit the provided regex match.
-
+
="X1") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -764,20 +764,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -827,54 +827,54 @@ Keep and drop vari
We can use the exact_match
argument to select a specific set of variables:
-
+
=["X1", "X2"], exact_match=True) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep
-
+
@@ -924,20 +924,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -987,54 +987,54 @@ Keep and drop vari
We can also easily drop variables via the drop
argument:
-
+
=["X1"]) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop
-
+
@@ -1075,20 +1075,20 @@ Keep and drop vari
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1141,54 +1141,54 @@ Keep and drop vari
Hide fixed effects or SE-type rows
We can hide the rows showing the relevant fixed effects and those showing the S.E. type by setting show_fe=False
and show_setype=False
(for instance when the set of fixed effects or the estimation method for the std. errors is the same for all models and you want to describe this in the text or table notes rather than displaying it in the table).
-
+
=False, show_se_type=False) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], show_fe
-
+
@@ -1283,54 +1283,54 @@ Hide fi
Display p-values or confidence intervals
By default, pf.etable()
reports standard errors. But we can also ask to output p-values or confidence intervals via the coef_fmt
function argument.
-
+
="b \n (se) \n [p]") pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt
-
+
@@ -1389,20 +1389,20 @@ D
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1455,54 +1455,54 @@ D
Significance levels and rounding
Additionally, we can also overwrite the defaults for the reported significance levels and control the rounding of results via the signif_code
and digits
function arguments:
-
+
=[0.01, 0.05, 0.1], digits=5) pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code
-
+
@@ -1561,20 +1561,20 @@ Significa
fe
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1627,7 +1627,7 @@ Significa
Other output formats
By default, pf.etable()
returns a GT object (see the Great Tables package), but you can also opt to dataframe, markdown, or latex output via the type
argument.
-
+
# Pandas styler output:
pf.etable(
@@ -1689,20 +1689,20 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
-0.04082 (0.08093)
- f2
- -
+ f1
x
x
- -
x
x
-
-
- f1
x
x
+
+
+ f2
+ -
x
x
+ -
x
x
@@ -1738,7 +1738,7 @@ Other output formats<
-
+
# Markdown output:
pf.etable(
@@ -1758,8 +1758,8 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
X1:X2 0.01057 -0.04082
(0.01818) (0.08093)
------------------------------------------------------------------------------------------------
-f2 - x x - x x
f1 x x x x x x
+f2 - x x - x x
------------------------------------------------------------------------------------------------
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
@@ -1769,7 +1769,7 @@ Other output formats<
To obtain latex output use format = "tex"
. If you want to save the table as a tex file, you can use the filename=
argument to specify the respective path where it should be saved. If you want the latex code to be displayed in the notebook, you can use the print_tex=True
argument. Etable will use latex packages booktabs
, threeparttable
and makecell
for the table layout, so don’t forget to include these packages in your latex document.
-
+
# LaTex output (include latex packages booktabs, threeparttable, and makecell in your document):
= pf.etable(
tab
@@ -1780,7 +1780,7 @@ [fit1, fit2, fit3, fit4, fit5, fit6],Other output formats<
)
The following code generates a pdf including the regression table which you can display clicking on the link below the cell:
-
+
## Use pylatex to create a tex file with the table
@@ -1814,7 +1814,7 @@ Other output formats<
Rename variables
You can also rename variables if you want to have a more readable output. Just pass a dictionary to the labels
argument. Note that interaction terms will also be relabeled using the specified labels for the interacted variables (if you want to manually relabel an interaction term differently, add it to the dictionary).
-
+