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
from great_tables import loc, style
from IPython.display import FileLink, display

import pyfixest as pf

%load_ext autoreload
%autoreload 2

data = pf.get_data()

fit1 = pf.feols("Y ~ X1 + X2 | f1", data=data)
fit2 = pf.feols("Y ~ X1 + X2 | f1 + f2", data=data)
fit3 = pf.feols("Y ~ X1 *X2 | f1 + f2", data=data)
fit4 = pf.feols("Y2 ~ X1 + X2 | f1", data=data)
fit5 = pf.feols("Y2 ~ X1 + X2 | f1 + f2", data=data)
fit6 = pf.feols("Y2 ~ X1 *X2 | f1 + f2", data=data)

Basic Usage

We can compare all regression models via the pyfixest-internal pf.etable() function:

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
X2 -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
X1:X2 0.011
(0.018)
-0.041
(0.081)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

You can also estimate and display multiple regressions with one line of code using the (py)fixest stepwise notation:

pf.etable(pf.feols("Y+Y2~csw(X1,X2,X1:X2)", data=data))
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -1.000***
(0.085)
-0.993***
(0.082)
-0.992***
(0.082)
-1.322***
(0.215)
-1.316***
(0.214)
-1.316***
(0.215)
X2 -0.176***
(0.022)
-0.197***
(0.036)
-0.133*
(0.057)
-0.132
(0.095)
X1:X2 0.020
(0.027)
-0.001
(0.071)
Intercept 0.919***
(0.112)
0.889***
(0.108)
0.888***
(0.108)
1.064***
(0.283)
1.042***
(0.283)
1.042***
(0.283)
stats
Observations 998 998 998 999 999 999
S.E. type iid iid iid iid iid iid
R2 0.123 0.177 0.177 0.037 0.042 0.042
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

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.

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep="X1")
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
X1:X2 0.011
(0.018)
-0.041
(0.081)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

We can use the exact_match argument to select a specific set of variables:

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], keep=["X1", "X2"], exact_match=True)
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
X2 -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

We can also easily drop variables via the drop argument:

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], drop=["X1"])
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X2 -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

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).

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], show_fe=False, show_se_type=False)
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
X2 -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
X1:X2 0.011
(0.018)
-0.041
(0.081)
stats
Observations 997 997 997 998 998 998
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

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.

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], coef_fmt="b \n (se) \n [p]")
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
[0.000]
-0.924***
(0.061)
[0.000]
-0.924***
(0.061)
[0.000]
-1.267***
(0.174)
[0.000]
-1.232***
(0.192)
[0.000]
-1.231***
(0.192)
[0.000]
X2 -0.174***
(0.018)
[0.000]
-0.174***
(0.015)
[0.000]
-0.185***
(0.025)
[0.000]
-0.131**
(0.042)
[0.005]
-0.118**
(0.042)
[0.008]
-0.074
(0.104)
[0.482]
X1:X2 0.011
(0.018)
[0.565]
-0.041
(0.081)
[0.618]
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error) [p-value]

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:

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], signif_code=[0.01, 0.05, 0.1], digits=5)
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.94953***
(0.06652)
-0.92405***
(0.06093)
-0.92417***
(0.06094)
-1.26655***
(0.17359)
-1.23153***
(0.19228)
-1.23100***
(0.19167)
X2 -0.17423***
(0.01840)
-0.17411***
(0.01461)
-0.18550***
(0.02516)
-0.13056***
(0.04239)
-0.11767***
(0.04152)
-0.07369
(0.10356)
X1:X2 0.01057
(0.01818)
-0.04082
(0.08093)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180
Significance levels: * p < 0.1, ** p < 0.05, *** p < 0.01. Format of coefficient cell: Coefficient (Std. Error)

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(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    signif_code=[0.01, 0.05, 0.1],
    digits=5,
    coef_fmt="b (se)",
    type="df",
)
est1 est2 est3 est4 est5 est6
depvar Y Y Y Y2 Y2 Y2
X1 -0.94953*** (0.06652) -0.92405*** (0.06093) -0.92417*** (0.06094) -1.26655*** (0.17359) -1.23153*** (0.19228) -1.23100*** (0.19167)
X2 -0.17423*** (0.01840) -0.17411*** (0.01461) -0.18550*** (0.02516) -0.13056*** (0.04239) -0.11767*** (0.04152) -0.07369 (0.10356)
X1:X2 0.01057 (0.01818) -0.04082 (0.08093)
f2 - x x - x x
f1 x x 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
R2 0.48899 0.65904 0.65916 0.12017 0.17151 0.17180
# Markdown output:
pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    signif_code=[0.01, 0.05, 0.1],
    digits=5,
    type="md",
)
index                 est1          est2          est3          est4          est5          est6
------------  ------------  ------------  ------------  ------------  ------------  ------------
depvar                   Y             Y             Y            Y2            Y2            Y2
------------------------------------------------------------------------------------------------
X1            -0.94953***   -0.92405***   -0.92417***   -1.26655***   -1.23153***   -1.23100***
                 (0.06652)     (0.06093)     (0.06094)     (0.17359)     (0.19228)     (0.19167)
X2            -0.17423***   -0.17411***   -0.18550***   -0.13056***   -0.11767***      -0.07369
                 (0.01840)     (0.01461)     (0.02516)     (0.04239)     (0.04152)     (0.10356)
X1:X2                                         0.01057                                  -0.04082
                                             (0.01818)                                 (0.08093)
------------------------------------------------------------------------------------------------
f2                       -             x             x             -             x             x
f1                       x             x             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
R2                 0.48899       0.65904       0.65916       0.12017       0.17151       0.17180
------------------------------------------------------------------------------------------------

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):
tab = pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    signif_code=[0.01, 0.05, 0.1],
    digits=2,
    type="tex",
    print_tex=True,
)

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


def make_pdf(tab, file):
    "Create a PDF document with tex table."
    doc = pl.Document()
    doc.packages.append(pl.Package("booktabs"))
    doc.packages.append(pl.Package("threeparttable"))
    doc.packages.append(pl.Package("makecell"))

    with (
        doc.create(pl.Section("A PyFixest LateX Table")),
        doc.create(pl.Table(position="htbp")) as table,
    ):
        table.append(pl.NoEscape(tab))

    doc.generate_pdf(file, clean_tex=False)


# Compile latex to pdf & display a button with the hyperlink to the pdf
# requires tex installation
run = False
if run:
    make_pdf(tab, "latexdocs/SampleTableDoc")
display(FileLink("latexdocs/SampleTableDoc.pdf"))

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).

labels = {
    "Y": "Wage",
    "Y2": "Wealth",
    "X1": "Age",
    "X2": "Years of Schooling",
    "f1": "Industry",
    "f2": "Year",
}

pf.etable([fit1, fit2, fit3, fit4, fit5, fit6], labels=labels)
Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - x x - x x
Industry x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

If 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.

pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    labels=labels,
    felabels={"f1": "Industry Fixed Effects", "f2": "Year Fixed Effects"},
)
Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year Fixed Effects - x x - x x
Industry Fixed Effects x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Custom model headlines

You can also add custom headers for each model by passing a list of strings to the model_headers argument.

pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    labels=labels,
    model_heads=["US", "China", "EU", "US", "China", "EU"],
)
  Wage Wealth
US China EU US China EU
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - x x - x x
Industry x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Or 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.

pf.etable(
    [fit1, fit4, fit2, fit5, fit3, fit6],
    labels=labels,
    model_heads=["US", "US", "China", "China", "EU", "EU"],
    head_order="hd",
)
  US China EU
Wage Wealth Wage Wealth Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-1.267***
(0.174)
-0.924***
(0.061)
-1.232***
(0.192)
-0.924***
(0.061)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.131**
(0.042)
-0.174***
(0.015)
-0.118**
(0.042)
-0.185***
(0.025)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - - x x x x
Industry x x x x x x
stats
Observations 997 998 997 998 997 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.120 0.659 0.172 0.659 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Remove the dependent variables from the headers:

pf.etable(
    [fit1, fit4, fit2, fit5, fit3, fit6],
    labels=labels,
    model_heads=["US", "US", "China", "China", "EU", "EU"],
    head_order="",
)
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-1.267***
(0.174)
-0.924***
(0.061)
-1.232***
(0.192)
-0.924***
(0.061)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.131**
(0.042)
-0.174***
(0.015)
-0.118**
(0.042)
-0.185***
(0.025)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - - x x x x
Industry x x x x x x
stats
Observations 997 998 997 998 997 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.120 0.659 0.172 0.659 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Further custom model information

You 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.

pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    labels=labels,
    custom_model_stats={
        "Number of Clusters": [42, 42, 42, 37, 37, 37],
        "Additional Info": ["A", "A", "B", "B", "C", "C"],
    },
)
Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - x x - x x
Industry x x x x x x
stats
Number of Clusters 42 42 42 37 37 37
Additional Info A A B B C C
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Custom table notes

You can replace the default table notes with your own notes using the notes argument.

mynotes = "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."
pf.etable(
    [fit1, fit4, fit2, fit5, fit3, fit6],
    labels=labels,
    model_heads=["US", "US", "China", "China", "EU", "EU"],
    head_order="hd",
    notes=mynotes,
)
  US China EU
Wage Wealth Wage Wealth Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-1.267***
(0.174)
-0.924***
(0.061)
-1.232***
(0.192)
-0.924***
(0.061)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.131**
(0.042)
-0.174***
(0.015)
-0.118**
(0.042)
-0.185***
(0.025)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year - - x x x x
Industry x x x x x x
stats
Observations 997 998 997 998 997 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.120 0.659 0.172 0.659 0.172
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.

Publication-ready LaTex tables

With few lines of code you thus obtain a publication-ready latex table:

tab = pf.etable(
    [fit1, fit4, fit2, fit5, fit3, fit6],
    labels=labels,
    model_heads=["US", "US", "China", "China", "EU", "EU"],
    head_order="hd",
    type="tex",
    notes=mynotes,
    show_fe=True,
    show_se_type=False,
    custom_model_stats={
        "Number of Clusters": [42, 42, 42, 37, 37, 37],
    },
)

# Compile latex to pdf & display a button with the hyperlink to the pdf
run = False
if run:
    make_pdf(tab, "latexdocs/SampleTableDoc2")
display(FileLink("latexdocs/SampleTableDoc2.pdf"))

Rendering Tables in Quarto

When you use quarto you can include latex tables generated by pyfixest when rendering the qmd file as pdf. Just specify output: asis in the code block options of the respective chunk and print the LaTex string returned by etable. Don’t forget to include the \usepackage commands for necessary latex packages in the YAML block. Here you find a sample qmd file.

When you render either a jupyter notebook or qmd file to html it is advisable to turn html-table-processing off in quarto as otherwise quarto adds further formatting which alters how your tables look like. You can do this in a raw cell at the top of your document.

---
format:
  html:
    html-table-processing: none
---

Descriptive Statistics via pf.dtable()

The function pf.dtable() allows to display descriptive statistics for a set of variables in the same layout.

Basic Usage of dtable

Specify the variables you want to display the descriptive statistics for. You can also use a dictionary to rename the variables and add a caption.

pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    labels=labels,
    caption="Descriptive statistics",
    digits=2,
)
Descriptive statistics
N Mean Std. Dev.
Wage 999 -0.13 2.30
Wealth 1000 -0.31 5.58
Age 999 1.04 0.81
Years of Schooling 1000 -0.13 3.05

Choose the set of statistics to be displayed with stats. You can use any pandas aggregation functions.

pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    stats=["count", "mean", "std", "min", "max"],
    labels=labels,
    caption="Descriptive statistics",
)
Descriptive statistics
N Mean Std. Dev. Min Max
Wage 999 -0.13 2.30 -6.54 6.91
Wealth 1000 -0.31 5.58 -16.97 17.16
Age 999 1.04 0.81 0.00 2.00
Years of Schooling 1000 -0.13 3.05 -9.67 10.99

Summarize by characteristics in columns and rows

You 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.

# Generate some categorial data
data["country"] = np.random.choice(["US", "EU"], data.shape[0])
data["occupation"] = np.random.choice(["Blue collar", "White collar"], data.shape[0])

# Drop nan values to have balanced data
data.dropna(inplace=True)

pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    labels=labels,
    bycol=["country", "occupation"],
    stats=["count", "mean", "std"],
    caption="Descriptive statistics",
    stats_labels={"count": "Number of observations"},
    counts_row_below=True,
)
Descriptive statistics
  EU US
Blue collar White collar Blue collar White collar
Mean Std. Dev. Mean Std. Dev. Mean Std. Dev. Mean Std. Dev.
stats
Wage -0.17 2.35 -0.13 2.27 -0.21 2.31 0.01 2.30
Wealth -0.33 5.74 -0.67 5.44 -0.09 5.68 -0.26 5.48
Age 1.03 0.81 1.05 0.82 1.03 0.81 1.07 0.79
Years of Schooling -0.15 3.10 -0.35 3.05 -0.06 3.06 0.01 2.99
nobs
Number of observations 241 221 291 244

You 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.

You 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).

pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    labels=labels,
    bycol=["country", "occupation"],
    stats=["mean_newline_std", "count"],
    caption="Descriptive statistics",
    stats_labels={"count": "Number of observations"},
    counts_row_below=True,
    hide_stats=True,
)
Descriptive statistics
EU US
Blue collar White collar Blue collar White collar
stats
Wage -0.17
(2.35)
-0.13
(2.27)
-0.21
(2.31)
0.01
(2.30)
Wealth -0.33
(5.74)
-0.67
(5.44)
-0.09
(5.68)
-0.26
(5.48)
Age 1.03
(0.81)
1.05
(0.82)
1.03
(0.81)
1.07
(0.79)
Years of Schooling -0.15
(3.10)
-0.35
(3.05)
-0.06
(3.06)
0.01
(2.99)
nobs
Number of observations 241 221 291 244
Note: Displayed statistics are Mean (Std. Dev.).

You 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).

pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    labels=labels,
    bycol=["country"],
    byrow="occupation",
    stats=["count", "mean", "std"],
    caption="Descriptive statistics",
)
Descriptive statistics
EU US
N Mean Std. Dev. N Mean Std. Dev.
Blue collar
Wage 241 -0.17 2.35 291 -0.21 2.31
Wealth 241 -0.33 5.74 291 -0.09 5.68
Age 241 1.03 0.81 291 1.03 0.81
Years of Schooling 241 -0.15 3.10 291 -0.06 3.06
White collar
Wage 221 -0.13 2.27 244 0.01 2.30
Wealth 221 -0.67 5.44 244 -0.26 5.48
Age 221 1.05 0.82 244 1.07 0.79
Years of Schooling 221 -0.35 3.05 244 0.01 2.99

And you can again export descriptive statistics tables also to LaTex:

dtab = pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    labels=labels,
    bycol=["country"],
    byrow="occupation",
    stats=["count", "mean", "std"],
    type="tex",
)

run = False
if run:
    make_pdf(dtab, "latexdocs/SampleTableDoc3")
display(FileLink("latexdocs/SampleTableDoc3.pdf"))

Table Layout for DataFrames with pf.make_table()

pf.make_table() is called by pf.etable() and pf.dtable() to generate the tables in “gt” and “tex” format. But you can also use it directly to generate tables in the same layout from other pandas dataframes.

Basic Usage of make_table

df = pd.DataFrame(np.random.randn(4, 4).round(2), columns=["A", "B", "C", "D"])

# Make Booktabs style table
pf.make_table(df=df, caption="This is a caption", notes="These are notes")
This is a caption
A B C D
0 0.47 -1.08 -1.92 -1.06
1 0.91 0.71 -1.41 -1.13
2 0.03 1.85 0.55 -0.22
3 -1.17 0.25 0.9 -1.07
These are notes

Mutiindex DataFrames

When 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.

# Create a multiindex dataframe with random data
row_index = pd.MultiIndex.from_tuples(
    [
        ("Group 1", "Variable 1"),
        ("Group 1", "Variable 2"),
        ("Group 1", "Variable 3"),
        ("Group 2", "Variable 4"),
        ("Group 2", "Variable 5"),
        ("Group 3", "Variable 6"),
    ]
)

col_index = pd.MultiIndex.from_product([["A", "B"], ["X", "Y"], ["High", "Low"]])
df = pd.DataFrame(np.random.randn(6, 8).round(3), index=row_index, columns=col_index)

pf.make_table(df=df, caption="This is a caption", notes="These are notes")
This is a caption
  A B
X Y X Y
High Low High Low High Low High Low
Group 1
Variable 1 -0.378 0.415 -0.78 -1.139 -0.077 0.618 -0.756 -0.952
Variable 2 0.446 -0.879 0.375 0.49 0.244 0.984 0.059 -2.679
Variable 3 0.053 0.8 0.758 1.637 -1.438 -1.403 -1.358 1.187
Group 2
Variable 4 -0.034 0.935 -0.917 0.83 0.03 0.226 -1.231 0.919
Variable 5 -0.641 -0.41 -0.961 -1.24 -0.117 -0.735 -0.68 0.107
Group 3
Variable 6 0.578 0.653 -0.103 0.238 1.815 -0.133 -0.458 0.649
These are notes

You 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.

pf.make_table(
    df=df, caption="This is a caption", notes="These are notes", rgroup_display=False
).tab_style(style=style.text(style="italic"), locations=loc.body(rows=[1, 5]))
This is a caption
  A B
X Y X Y
High Low High Low High Low High Low
Group 1
Variable 1 -0.378 0.415 -0.78 -1.139 -0.077 0.618 -0.756 -0.952
Variable 2 0.446 -0.879 0.375 0.49 0.244 0.984 0.059 -2.679
Variable 3 0.053 0.8 0.758 1.637 -1.438 -1.403 -1.358 1.187
Group 2
Variable 4 -0.034 0.935 -0.917 0.83 0.03 0.226 -1.231 0.919
Variable 5 -0.641 -0.41 -0.961 -1.24 -0.117 -0.735 -0.68 0.107
Group 3
Variable 6 0.578 0.653 -0.103 0.238 1.815 -0.133 -0.458 0.649
These are notes

Custom Styling with Great Tables

You can use the rich set of methods offered by Great Tables to further customize the table display when the type is “gt”.

Example Styling

(
    pf.etable([fit1, fit2, fit3, fit4, fit5, fit6])
    .tab_options(
        column_labels_background_color="cornsilk",
        stub_background_color="whitesmoke",
    )
    .tab_style(
        style=style.fill(color="mistyrose"),
        locations=loc.body(columns="(3)", rows=["X2"]),
    )
)
Y Y2
(1) (2) (3) (4) (5) (6)
coef
X1 -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
X2 -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
X1:X2 0.011
(0.018)
-0.041
(0.081)
fe
f2 - x x - x x
f1 x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)

Defining Table Styles: Some Examples

You 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).

style_print = {
    "table_font_size": "12px",
    "heading_title_font_size": "12px",
    "source_notes_font_size": "8px",
    "data_row_padding": "3px",
    "column_labels_padding": "3px",
    "row_group_border_top_style": "hidden",
    "table_body_border_top_style": "None",
    "table_body_border_bottom_width": "1px",
    "column_labels_border_top_width": "1px",
    "table_width": "14cm",
}


style_presentation = {
    "table_font_size": "16px",
    "table_font_color_light": "white",
    "table_body_border_top_style": "hidden",
    "table_body_border_bottom_style": "hidden",
    "heading_title_font_size": "18px",
    "source_notes_font_size": "12px",
    "data_row_padding": "3px",
    "column_labels_padding": "6px",
    "column_labels_background_color": "midnightblue",
    "stub_background_color": "whitesmoke",
    "row_group_background_color": "whitesmoke",
    "table_background_color": "whitesmoke",
    "heading_background_color": "white",
    "source_notes_background_color": "white",
    "column_labels_border_bottom_color": "white",
    "column_labels_font_weight": "bold",
    "row_group_font_weight": "bold",
    "table_width": "18cm",
}
t1 = pf.dtable(
    data,
    vars=["Y", "Y2", "X1", "X2"],
    stats=["count", "mean", "std", "min", "max"],
    labels=labels,
    caption="Descriptive statistics",
)

t2 = pf.etable(
    [fit1, fit2, fit3, fit4, fit5, fit6],
    labels=labels,
    show_se=False,
    felabels={"f1": "Industry Fixed Effects", "f2": "Year Fixed Effects"},
    caption="Regression results",
)
display(t1.tab_options(**style_print))
display(t2.tab_options(**style_print))
Descriptive statistics
N Mean Std. Dev. Min Max
Wage 997 -0.13 2.31 -6.54 6.91
Wealth 997 -0.32 5.59 -16.97 17.16
Age 997 1.04 0.81 0.00 2.00
Years of Schooling 997 -0.13 3.05 -9.67 10.99
Regression results
Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year Fixed Effects - x x - x x
Industry Fixed Effects x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)
style_printDouble = {
    "table_font_size": "12px",
    "heading_title_font_size": "12px",
    "source_notes_font_size": "8px",
    "data_row_padding": "3px",
    "column_labels_padding": "3px",
    "table_body_border_bottom_style": "double",
    "column_labels_border_top_style": "double",
    "column_labels_border_bottom_width": "0.5px",
    "row_group_border_top_style": "hidden",
    "table_body_border_top_style": "None",
    "table_width": "14cm",
}
display(t1.tab_options(**style_printDouble))
display(t2.tab_options(**style_printDouble))
Descriptive statistics
N Mean Std. Dev. Min Max
Wage 997 -0.13 2.31 -6.54 6.91
Wealth 997 -0.32 5.59 -16.97 17.16
Age 997 1.04 0.81 0.00 2.00
Years of Schooling 997 -0.13 3.05 -9.67 10.99
Regression results
Wage Wealth
(1) (2) (3) (4) (5) (6)
coef
Age -0.950***
(0.067)
-0.924***
(0.061)
-0.924***
(0.061)
-1.267***
(0.174)
-1.232***
(0.192)
-1.231***
(0.192)
Years of Schooling -0.174***
(0.018)
-0.174***
(0.015)
-0.185***
(0.025)
-0.131**
(0.042)
-0.118**
(0.042)
-0.074
(0.104)
Age × Years of Schooling 0.011
(0.018)
-0.041
(0.081)
fe
Year Fixed Effects - x x - x x
Industry Fixed Effects x x x x x x
stats
Observations 997 997 997 998 998 998
S.E. type by: f1 by: f1 by: f1 by: f1 by: f1 by: f1
R2 0.489 0.659 0.659 0.120 0.172 0.172
Significance levels: * p < 0.05, ** p < 0.01, *** p < 0.001. Format of coefficient cell: Coefficient (Std. Error)