The goal of this vignette is to show how you can start and maintain a new multilingual Quarto project using {babelquarto}. There are two types of projects: a book or a website. We will look at each type separately below.
If you want to turn an existing project into a multilingual project, have a look at vignette("convert").
Installing babelquarto
Before you can start a new multilingual project, you need to install {babelquarto}.
install.packages('babelquarto', repos = c('https://ropensci.r-universe.dev', 'https://cloud.r-project.org'))
Or from GitHub with:
# install.packages("pak")
pak::pak("ropensci-review-tools/babelquarto")
Starting a multilingual book
To start a multilingual book, use quarto_multilingual_book() with the parent_dir argument to specify where you want to create the project and the project_dir argument to specify the name of the project. The argument main_language is used to specify the main language of the project and further_languages lists all additional languages.
parent_dir <- withr::local_tempdir()
project_dir <- "multilingual_book"
babelquarto::quarto_multilingual_book(
  parent_dir = parent_dir,
  project_dir = project_dir,
  main_language = "en",
  further_languages = c("es", "fr")
)
[32m✔[39m Added configuration for en to [34m[34mconfig_path[34m[39m.
 
[32m✔[39m Added configuration for usees, fr to [34m[34mconfig_path[34m[39m.
 
 
Look at the _quarto.yml file in the project directory. To get familiar with the configuration, take a look at the example below:
project:
  type: book
book:
  site-url: https://example.com
  title: "multilingual_book"
  author: "Firstname Lastname"
  date: "10/21/2025"
  chapters:
    - index.qmd
    - intro.qmd
    - summary.qmd
    - references.qmd
bibliography: references.bib
format:
  html:
    theme:
      - cosmo
      - brand
babelquarto:
  languagelinks: sidebar
  languagecodes:
  - name: es
    text: "Version in es"
  - name: fr
    text: "Version in fr"
  - name: en
    text: "Version in en"
  mainlanguage: 'en'
  languages: ['es', 'fr']
title-es: title in es
title-fr: title in fr
description-es: description in es
description-fr: description in fr
author-es: author in es
author-fr: author in fr
lang: en
 
The file structure of the project looks like this:
[01;34m/tmp/Rtmpk3Zt4z/file83337ef9655/multilingual_book[0m
├── _quarto.yml
├── [01;35mcover.png[0m
├── index.es.qmd
├── index.fr.qmd
├── index.qmd
├── intro.es.qmd
├── intro.fr.qmd
├── intro.qmd
├── references.bib
├── references.es.qmd
├── references.fr.qmd
├── references.qmd
├── summary.es.qmd
├── summary.fr.qmd
└── summary.qmd
 
 
Each Quarto file has a Spanish and a French version. These files aren’t automatically translated and are just copies of the original English version. You will have to provide the translations yourself, or look at {babeldown} for automatic translation. If you look at the index.qmd file, you’ll see that the French file is called index.fr.qmd and the Spanish file is called index.es.qmd.
When you’re ready to render your book, use render_book(). From within a directory, you don’t need to specify the project_path argument as it defaults to ..
We end up with three books, that cross-link to each other from the left sidebar. Example.
By default, in interactive sessions a preview of the project will be opened, using the servr package. See the Previewing your multilingual project section for further details.
Starting a multilingual website
To start a multilingual website, use quarto_multilingual_website() with the parent_dir argument to specify where you want to create the project and the project_dir argument to specify the name of the project. The argument main_language is used to specify the main language of the project and further_languages lists all additional languages.
parent_dir <- withr::local_tempdir()
project_dir <- "multilingual_website"
babelquarto::quarto_multilingual_website(
  parent_dir = parent_dir,
  project_dir = project_dir,
  main_language = "en",
  further_languages = c("es", "fr")
)
[32m✔[39m Added configuration for en to [34m[34mconfig_path[34m[39m.
 
[32m✔[39m Added configuration for usees, fr to [34m[34mconfig_path[34m[39m.
 
 
Look at the _quarto.yml file in the project directory. To get familiar with the configuration, take a look at the example below:
project:
  type: website
website:
  site-url: https://example.com
  title: "multilingual_website"
  navbar:
    left:
      - href: index.qmd
        text: Home
      - about.qmd
format:
  html:
    theme:
      - cosmo
      - brand
    css: styles.css
    toc: true
babelquarto:
  languagelinks: navbar
  languagecodes:
  - name: es
    text: "Version in es"
  - name: fr
    text: "Version in fr"
  - name: en
    text: "Version in en"
  mainlanguage: 'en'
  languages: ['es', 'fr']
title-es: title in es
title-fr: title in fr
description-es: description in es
description-fr: description in fr
author-es: author in es
author-fr: author in fr
lang: en
 
Note that the fields like title-es, description-es, author-es are not indented so you could choose to cut and paste them below the default language’s title, description, author to more easily remember to update them if needed.
The file structure of the project looks like this:
[01;34m/tmp/Rtmpk3Zt4z/file8334c129295/multilingual_website[0m
├── _quarto.yml
├── about.es.qmd
├── about.fr.qmd
├── about.qmd
├── index.es.qmd
├── index.fr.qmd
├── index.qmd
└── styles.css
 
 
Each Quarto file has a Spanish and a French version. These files aren’t automatically translated and are just copies of the original English version. You will have to provide the translations yourself, or look at {babeldown} for automatic translation. If you look at the index.qmd file, you’ll see that the French file is called index.fr.qmd and the Spanish file is called index.es.qmd.
When you’re ready to render your website, use babelquarto::render_website():
We end up with a multilingual website. Example, source
Note that the _quarto.yml file created in this process has an example entry set for site-url:. Depending on how your multilingual project is being deployed, you may need to either remove or update this entry or alternatively, set the url during rendering by using the site_url argument in the render functions (e.g., babelquarto::render_website(site_url = "https://your.blog.net/your-multilingual-book/")). You can set the site-url field to your project’s URL directly by using the site_url parameter of quarto_multilingual_website()/quarto_multilingual_book().
Previewing your multilingual project
Once you have rendered your project, you will have a _site or _book folder in your project. In Quarto you would use quarto preview to be able to get a look at what you project looks like. Because of the way {babelquarto} operates, this isn’t possible. You can however preview your files using the {servr} package.
You can use servr::httw() to preview your project.
# For a multilingual website
servr::httw("_site")
# For a multilingual book
servr::httw("_book")
 
This will show an URL that you can open in your IDE or browser to see your project.
If the publishing source for your multilingual project is a ‘docs’ folder in your repository, you can preview the output with servr::httw("docs").
Publishing your project
Your Quarto multilingual website or book is a static website. To publish it, you need to deploy the output folder, i.e. _site or _book. That content can be served on any service, for instance GitHub Pages or Netlify.
Next steps
Take a deeper dive into the configuration options available in {babelquarto} and have a look at vignette("configuration").
If you want to translate your multilingual project using automatic translation with DeepL, you should have a look at babeldown.
If you want to setup your own CI and deploy your website, have a look at vignette("render-with-ci").
If you need to personalize the Quarto templates, have a look at vignette("custom-templates").