- 
                Notifications
    You must be signed in to change notification settings 
- Fork 33
Fonts #1039
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
          
     Open
      
        
      
            florian-lefebvre
  wants to merge
  47
  commits into
  main
  
    
      
        
          
  
    
      Choose a base branch
      
     
    
      
        
      
      
        
          
          
        
        
          
            
              
              
              
  
           
        
        
          
            
              
              
           
        
       
     
  
        
          
            
          
            
          
        
       
    
      
from
rfc/fonts
  
      
      
   
  
    
  
  
  
 
  
      
    base: main
Could not load branches
            
              
  
    Branch not found: {{ refName }}
  
            
                
      Loading
              
            Could not load tags
            
            
              Nothing to show
            
              
  
            
                
      Loading
              
            Are you sure you want to change the base?
            Some commits from the old base branch may be removed from the timeline,
            and old review comments may become outdated.
          
          
  
     Open
                    Fonts #1039
Changes from 16 commits
      Commits
    
    
            Show all changes
          
          
            47 commits
          
        
        Select commit
          Hold shift + click to select a range
      
      526ab7c
              
                feat: work on rfc
              
              
                florian-lefebvre c53956c
              
                feat: copy from stage 2
              
              
                florian-lefebvre 4b67e1f
              
                chore: raw ideas
              
              
                florian-lefebvre 5006242
              
                feat: examples
              
              
                florian-lefebvre 28b1c1c
              
                fix: link
              
              
                florian-lefebvre 987d13a
              
                chore: add todos
              
              
                florian-lefebvre c589609
              
                feat: work on providers
              
              
                florian-lefebvre 8250880
              
                feat: defaults and families
              
              
                florian-lefebvre 7233fbe
              
                feat: component and usage
              
              
                florian-lefebvre ff39e16
              
                fix: headings
              
              
                florian-lefebvre df5bbd5
              
                feat
              
              
                florian-lefebvre c79c1fa
              
                feat: drawbacks, alternatives, adoption
              
              
                florian-lefebvre caa649b
              
                feat: family
              
              
                florian-lefebvre 64b9ba4
              
                feat: tweak
              
              
                florian-lefebvre 94b5805
              
                Apply suggestions from code review
              
              
                florian-lefebvre 15e938a
              
                Update 0052-fonts.md
              
              
                florian-lefebvre 45bff8a
              
                feat: remove configurable defaults
              
              
                florian-lefebvre f238d52
              
                Update 0052-fonts.md
              
              
                florian-lefebvre 6454ba5
              
                feat: update local provider shape
              
              
                florian-lefebvre 6ce971c
              
                feat: caching
              
              
                florian-lefebvre 3c26829
              
                feat: add fallbacks
              
              
                florian-lefebvre a635c71
              
                feat: explain fallbacks
              
              
                florian-lefebvre 8e4e172
              
                feat: address reviews
              
              
                florian-lefebvre 6554d68
              
                fix: typo
              
              
                florian-lefebvre 8ab8bc0
              
                feat: update conditions order
              
              
                florian-lefebvre bb51b9d
              
                feat: remove cssVar prop
              
              
                florian-lefebvre 85dd3c2
              
                feat: as prop
              
              
                florian-lefebvre c4b13e6
              
                feat: update fontaine to capsize
              
              
                florian-lefebvre 4c97a06
              
                feat: update defaults
              
              
                florian-lefebvre 95c04c4
              
                feat: as prop
              
              
                florian-lefebvre f039f55
              
                chore: remove todo
              
              
                florian-lefebvre 64e71d1
              
                feat: disable automatic fallback generation
              
              
                florian-lefebvre 26a771d
              
                feat: update rfc
              
              
                florian-lefebvre 28eac1e
              
                Update 0052-fonts.md
              
              
                florian-lefebvre 360dc8f
              
                feat: cssVariable
              
              
                florian-lefebvre a057f65
              
                feat: update local family shape
              
              
                florian-lefebvre 19872a1
              
                feat: better local src
              
              
                florian-lefebvre 6ed0d41
              
                feat: clarify entrypoints
              
              
                florian-lefebvre 06de5b8
              
                feat: no default provider
              
              
                florian-lefebvre 265452b
              
                feat: renames
              
              
                florian-lefebvre a55738d
              
                feat: weight/style inference for local provider
              
              
                florian-lefebvre 8665236
              
                Merge branch 'main' into rfc/fonts
              
              
                florian-lefebvre 9eca2f9
              
                chore: reorder
              
              
                florian-lefebvre a2e5cd4
              
                feat: mention csp
              
              
                florian-lefebvre fb07e6f
              
                Update 0055-fonts.md
              
              
                florian-lefebvre 75b5810
              
                feat: getFontData()
              
              
                florian-lefebvre 72b6b46
              
                feat: granular preloads
              
              
                florian-lefebvre File filter
Filter by extension
Conversations
          Failed to load comments.   
        
        
          
      Loading
        
  Jump to
        
          Jump to file
        
      
      
          Failed to load files.   
        
        
          
      Loading
        
  Diff view
Diff view
There are no files selected for viewing
            File renamed without changes.
          
    
  
    
      This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
      Learn more about bidirectional Unicode characters
    
  
  
    
              | Original file line number | Diff line number | Diff line change | 
|---|---|---|
| @@ -0,0 +1,369 @@ | ||
| **If you have feedback and the feature is released as experimental, please leave it on the Stage 3 PR. Otherwise, comment on the Stage 2 issue (links below).** | ||
|  | ||
| - Start Date: 2024-10-15 | ||
| - Reference Issues: <!-- related issues, otherwise leave empty --> | ||
| - Implementation PR: <!-- leave empty --> | ||
| - Stage 2 Issue: https://github.com/withastro/roadmap/issues/1037 | ||
| - Stage 3 PR: https://github.com/withastro/roadmap/pull/1039 | ||
|  | ||
| # Summary | ||
|  | ||
| Have first-party support for fonts in Astro. | ||
|  | ||
| # Example | ||
|  | ||
| ```js | ||
| // astro config | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: ["Roboto", "Lato"], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| ```astro | ||
| --- | ||
| // layouts/Layout.astro | ||
| import { Font } from 'astro:fonts' | ||
| --- | ||
| <head> | ||
| <Font family='Inter' preload /> | ||
| <Font family='Lato' /> | ||
| <style> | ||
| h1 { | ||
| font-family: var(--astro-font-inter); | ||
| } | ||
| p { | ||
| font-family: var(--astro-font-lato); | ||
| } | ||
| </style> | ||
| </head> | ||
| ``` | ||
|  | ||
| # Background & Motivation | ||
|  | ||
| Fonts is one of those basic things when making a website, but also an annoying one to deal with. Should I just use a link to a remote font? Or download it locally? How should I handle preloads then? | ||
|  | ||
| The goal is to improve the DX around using fonts in Astro. | ||
|  | ||
| > Why not using fontsource? | ||
|  | ||
| Fontsource is great! But it's not intuitive to preload, and more importantly, doesn't have all fonts. The goal is to have a more generic API for fonts (eg. you want to use a paid provider like adobe). | ||
|  | ||
| # Goals | ||
|  | ||
| - Specify what font to use | ||
| - Cache fonts | ||
| - Specify what provider to use | ||
|         
                  matthewp marked this conversation as resolved.
              Show resolved
            Hide resolved | ||
| - Load/preload font on a font basis | ||
| - Generate fallbacks automatically | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| - Performant defaults | ||
| - Runtime agnostic | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| - Configure font families (subset, unicode range, weights etc) | ||
|  | ||
| # Non-Goals | ||
|  | ||
| - Runtime API (SSR is supported tho) | ||
| - Automatic subsetting (eg. analyzing static content) | ||
| - Automatic font detection (ie. downloading fonts based on font families names used in the user's project) | ||
|  | ||
| # Detailed Design | ||
|  | ||
| ## Astro config | ||
|  | ||
| ### Overview | ||
|  | ||
| The goal is to have a config that starts really simple for basic usecases, but can also be complex for advanced usecases. Here's an example of basic config: | ||
|  | ||
| ```js | ||
| import { defineConfig } from "astro/config"; | ||
|  | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: ["Roboto", "Lato"], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| That would get fonts from [Google Fonts](https://fonts.google.com/) with sensible defaults. | ||
|  | ||
| Here's a more complex example: | ||
|  | ||
| ```js | ||
| import { defineConfig, fontProviders } from "astro/config"; | ||
| import { myCustomFontProvider } from "./provider"; | ||
|  | ||
| export default defineConfig({ | ||
| fonts: { | ||
| providers: [ | ||
| fontProviders.adobe({ apiKey: process.env.ADOBE_FONTS_API_KEY }), | ||
| myCustomFontProvider(), | ||
| ], | ||
| defaults: { | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| provider: "adobe", | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| weights: [200, 700], | ||
| styles: ["italic"], | ||
| subsets: [ | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| "cyrillic-ext", | ||
| "cyrillic", | ||
| "greek-ext", | ||
| "greek", | ||
| "vietnamese", | ||
| "latin-ext", | ||
| "latin", | ||
| ], | ||
| }, | ||
| families: [ | ||
| "Roboto", | ||
| { | ||
| name: "Lato", | ||
| provider: "google", | ||
| weights: [100, 200, 300], | ||
| }, | ||
| { | ||
| name: "Custom", | ||
| provider: "local", | ||
| src: ["./assets/fonts/Custom.woff2"], | ||
| }, | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| ### Providers | ||
|  | ||
| #### Definition | ||
|  | ||
| A provider allows to retrieve font faces data from a font family name from a given CDN or abstraction. It's a [unifont](https://github.com/unjs/unifont) provider. | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
|  | ||
| #### Built-in providers | ||
|  | ||
|  | ||
| This is the default, and it's not configurable. Given the amount of fonts it supports by, it sounds like a logic choice. Note that the default can be customized for more advanced usecases. | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: ["Roboto"], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: { | ||
| defaults: { | ||
| provider: "local", | ||
| }, | ||
| families: [ | ||
| { | ||
| name: "Roboto", | ||
| provider: "google", | ||
| }, | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| ##### Local | ||
|  | ||
| This provider, unlike all the others, requires paths to fonts relatively to the root. | ||
|  | ||
| ```js | ||
| import { defineConfig, fontProviders } from "astro/config"; | ||
|  | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: [ | ||
| { | ||
| name: "Custom", | ||
| provider: "local", | ||
| src: ["./assets/fonts/Custom.woff2"], | ||
| }, | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| #### Opt-in providers | ||
|  | ||
| Other unifont providers are exported from `astro/config`. | ||
|  | ||
| ```js | ||
| import { defineConfig, fontProviders } from "astro/config"; | ||
|  | ||
| export default defineConfig({ | ||
| fonts: { | ||
| providers: [ | ||
| fontProviders.adobe({ apiKey: process.env.ADOBE_FONTS_API_KEY }), | ||
| ], | ||
| // ... | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| #### Why this API? | ||
|  | ||
| 1. **Coherent API**: a few things in Astro are using this pattern, namely integrations and vite plugins. It's simple to author as a library author, easy to use as a user | ||
|         
                  ematipico marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| 2. **Keep opt-in providers**: allows to only use 2 providers by default, and keeps the API open to anyone | ||
| 3. **Types!**: now that `defineConfig` [supports generics](https://github.com/withastro/astro/pull/12243), we can do powerful things! Associated with type generation, we can generate types for `families` `name`, infer the provider type from `defaults.provider` and more. | ||
|  | ||
| ### Defaults | ||
|  | ||
| Astro must provide sensible defaults when it comes to font weights, subsets and more. But when dealing with more custom advanced setups, it makes sense to be able to customize those defaults. They can be set in `fonts.defaults` and will be merged with Astro defaults. | ||
|  | ||
| We need to decide what default to provide. I can see 2 paths: | ||
|  | ||
| | Path | Example (weight) | Advantage | Downside | | ||
| | --------- | ------------------- | --------------------- | --------------------------------------------------------------------------------- | | ||
| | Minimal | Only include `400` | Lightweight | People will probably struggle by expecting all weights to be available by default | | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
| | Extensive | Include all weights | Predictable for users | Heavier by default | | ||
|  | ||
| ### Families | ||
|  | ||
| A family is made of a least a `name`: | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: [ | ||
| { | ||
| name: "Roboto", | ||
| }, | ||
| "Roboto", // Shorthand | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| It can specify options such as `weights`, `subsets` that default to the value of `fonts.defaults`: | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: [ | ||
| { | ||
| name: "Roboto", | ||
| weights: [400, 600], | ||
| }, | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| It can also specify a `provider` (and `src` if it's the `local` provider): | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: { | ||
| families: [ | ||
| { | ||
| name: "Roboto", | ||
| provider: "local", | ||
| src: "./Roboto.woff2", | ||
| }, | ||
| ], | ||
| }, | ||
| }); | ||
| ``` | ||
|  | ||
| ### Font component | ||
|  | ||
| Setting the config (see above) configures what fonts to download, but it doesn't include font automatically on pages. Instead, we provide a `<Font />` component that can be used to compose where and how to load fonts. | ||
|  | ||
| ```astro | ||
| --- | ||
| import { Font } from "astro:assets" | ||
| --- | ||
| <head> | ||
| <Font family="Inter" preload cssVar="primary-font" /> | ||
| <Font family="Lato" /> | ||
| </head> | ||
| ``` | ||
|  | ||
| ### Family | ||
|  | ||
| The family will be typed using type gen, based on the user's config. | ||
|  | ||
| ### Preload | ||
|  | ||
| Defaults to `false`: | ||
|  | ||
| - **Enabled**: Outputs a preload link tag and a style tag, without fallbacks | ||
| - **Disabled**: Output a style tag with fallbacks (generated using [fontaine](https://github.com/unjs/fontaine)) | ||
|  | ||
| ### cssVar | ||
|  | ||
| Defaults to `astro-font-${computedFontName}`. Specifies what identifier to use for the generated css variable. This is useful for font families names that may contain special character or conflict with other fonts. | ||
|  | ||
| ## Usage | ||
|  | ||
| Since fallbacks may be generated for a given family name, this name can't be used alone reliably: | ||
|  | ||
| ```css | ||
| h1 { | ||
| font-family: "Inter"; /* Should actually be "Inter", "Inter Fallback" */ | ||
| } | ||
| ``` | ||
|  | ||
| To solve this issue, a css variable is provided by the style tage generated by the `<Font />` component: | ||
|  | ||
| ```css | ||
| h1 { | ||
| font-family: var(--astro-font-inter); /* "Inter", "Inter Fallback" */ | ||
| } | ||
| ``` | ||
|  | ||
| ## How it works under the hood | ||
|  | ||
| - Once the config is fully resolved, we get fonts face data using `unifont` | ||
| - We generate fallbacks using `fontaine` and pass all the data we need through a virtual import, used by the `<Font />` component | ||
| - We inject a vite middleware in development to download fonts as they are requested in development | ||
| - During build, we download all fonts and put them in `outDir` | ||
|         
                  florian-lefebvre marked this conversation as resolved.
              Outdated
          
            Show resolved
            Hide resolved | ||
|  | ||
| Data is cached to `cacheDir` for builds and `.astro/fonts` in development. | ||
|  | ||
| # Testing Strategy | ||
|  | ||
| - Integration tests | ||
| - Experimental flag (`experimental.fonts`) | ||
|  | ||
| # Drawbacks | ||
|  | ||
| I have not identified any outstanding drawback: | ||
|  | ||
| - **Implementation cost, both in term of code size and complexity**: fine | ||
| - **Whether the proposed feature can be implemented in user space**: yes | ||
| - **Impact on teaching people Astro**: should make things easier, will need updating docs | ||
| - **Integration of this feature with other existing and planned features**: reuses `astro:assets` to export the component, otherwise isolated from other features | ||
| - **Is it a breaking change?** No | ||
|  | ||
| # Alternatives | ||
|  | ||
| ## As an integration | ||
|  | ||
| This feature could be developed as an integration, eg. `@astrojs/fonts`. It will probably be an internal integration (like actions) but making it part of core allows to make it more discoverable, more used. It also allows to use the `astro:assets` module. | ||
|  | ||
| ## Different API for simpler cases | ||
|  | ||
| The following API has been suggested for the simpler cases: | ||
|  | ||
| ```js | ||
| export default defineConfig({ | ||
| fonts: ["Roboto"], | ||
| }); | ||
| ``` | ||
|  | ||
| I'd love to support such API where you can provide fonts top level, or inside `fonts.families` but we can't. We can't because of how the integration API `defineConfig()` works. What if a user provides fonts names as `fonts`, and an integration provides fonts names as `fonts.families`? Given how the merging works, the shape of `AstroUserConfig` and `AstroConfig` musn't be too different. It already caused issues with i18n in the past. | ||
|  | ||
| # Adoption strategy | ||
|  | ||
| - **If we implement this proposal, how will existing Astro developers adopt it?** Fonts setups can vary a lot but migrating to the core fonts api should not require too much work | ||
| - **Is this a breaking change? Can we write a codemod?** No | ||
| - **How will this affect other projects in the Astro ecosystem?** This should make [`astro-font`](https://github.com/rishi-raj-jain/astro-font) obsolete | ||
|  | ||
| # Unresolved Questions | ||
|  | ||
| - We need to see how merging `fonts.defaults` will work, especially for `updateConfig()`. Should we merge arrays in this case? | ||
| - We need to check if fallbacks should still be included for preloaded fonts | ||
  Add this suggestion to a batch that can be applied as a single commit.
  This suggestion is invalid because no changes were made to the code.
  Suggestions cannot be applied while the pull request is closed.
  Suggestions cannot be applied while viewing a subset of changes.
  Only one suggestion per line can be applied in a batch.
  Add this suggestion to a batch that can be applied as a single commit.
  Applying suggestions on deleted lines is not supported.
  You must change the existing code in this line in order to create a valid suggestion.
  Outdated suggestions cannot be applied.
  This suggestion has been applied or marked resolved.
  Suggestions cannot be applied from pending reviews.
  Suggestions cannot be applied on multi-line comments.
  Suggestions cannot be applied while the pull request is queued to merge.
  Suggestion cannot be applied right now. Please check back later.
  
    
  
    
Uh oh!
There was an error while loading. Please reload this page.