From 6219a6d7005e9e23fc622ac653db57265b660678 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Tue, 3 Sep 2024 17:10:42 -0300 Subject: [PATCH 01/28] add editor font face resolver component --- .../src/components/block-canvas/index.js | 3 + .../components/editor-fonts-resolver/index.js | 63 ++++++++ .../components/editor-fonts-resolver/utils.js | 147 ++++++++++++++++++ 3 files changed, 213 insertions(+) create mode 100644 packages/block-editor/src/components/editor-fonts-resolver/index.js create mode 100644 packages/block-editor/src/components/editor-fonts-resolver/utils.js diff --git a/packages/block-editor/src/components/block-canvas/index.js b/packages/block-editor/src/components/block-canvas/index.js index 9bad3ca22e95f..03f3a4f2eb5f1 100644 --- a/packages/block-editor/src/components/block-canvas/index.js +++ b/packages/block-editor/src/components/block-canvas/index.js @@ -15,6 +15,7 @@ import WritingFlow from '../writing-flow'; import { useMouseMoveTypingReset } from '../observe-typing'; import { useBlockSelectionClearer } from '../block-selection-clearer'; import { useBlockCommands } from '../use-block-commands'; +import EditorFontsResolver from '../editor-fonts-resolver'; // EditorStyles is a memoized component, so avoid passing a new // object reference on each render. @@ -43,6 +44,7 @@ export function ExperimentalBlockCanvas( { __unstableContentRef={ localRef } style={ { height, display: 'flex' } } > + + { children } diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/editor-fonts-resolver/index.js new file mode 100644 index 0000000000000..17fd238c39a8c --- /dev/null +++ b/packages/block-editor/src/components/editor-fonts-resolver/index.js @@ -0,0 +1,63 @@ +/** + * WordPress dependencies + */ +import { useState, useEffect, useMemo } from '@wordpress/element'; +import { useSelect } from '@wordpress/data'; + +/** + * Internal dependencies + */ +import { useGlobalSetting } from '../global-styles/hooks'; +import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; + +function EditorFontsResolver() { + const [ loadedFontUrls ] = useState( new Set() ); + + const { currentTheme } = useSelect( ( select ) => { + return { + currentTheme: + // Using 'core' as string to avoid circular dependency importing from @wordpress/core-data + select( 'core' ).getCurrentTheme(), // eslint-disable-line @wordpress/data-no-store-string-literals + }; + }, [] ); + + //Get the fonts from merged theme.json settings.fontFamilies + const [ fontFamilies ] = useGlobalSetting( 'typography.fontFamilies' ); + + const fontFaces = useMemo( () => { + return Object.values( fontFamilies ) + .flat() + .map( ( family ) => family.fontFace ) + .filter( Boolean ) + .flat(); + }, [ fontFamilies ] ); + + const loadFontFaceAsset = async ( fontFace ) => { + // If the font doesn't have a src, don't load it. + if ( ! fontFace.src ) { + return; + } + // Get the src of the font. + const src = getDisplaySrcFromFontFace( + fontFace.src, + currentTheme?.stylesheet_uri + ); + // If the font is already loaded, don't load it again. + if ( ! src || loadedFontUrls.has( src ) ) { + return; + } + // Load the font in the browser. + loadFontFaceInBrowser( fontFace, src ); + // Add the font to the loaded fonts list. + loadedFontUrls.add( src ); + }; + + useEffect( () => { + fontFaces.forEach( loadFontFaceAsset ); + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [ fontFaces ] ); + + return null; +} + +export default EditorFontsResolver; diff --git a/packages/block-editor/src/components/editor-fonts-resolver/utils.js b/packages/block-editor/src/components/editor-fonts-resolver/utils.js new file mode 100644 index 0000000000000..23f22a45cc308 --- /dev/null +++ b/packages/block-editor/src/components/editor-fonts-resolver/utils.js @@ -0,0 +1,147 @@ +/* + * Format the font face name to use in the font-family property of a font face. + * + * The input can be a string with the font face name or a string with multiple font face names separated by commas. + * It removes the leading and trailing quotes from the font face name. + * + * @param {string} input - The font face name. + * @return {string} The formatted font face name. + * + * Example: + * formatFontFaceName("Open Sans") => "Open Sans" + * formatFontFaceName("'Open Sans', sans-serif") => "Open Sans" + * formatFontFaceName(", 'Open Sans', 'Helvetica Neue', sans-serif") => "Open Sans" + */ +export function formatFontFaceName( input ) { + if ( ! input ) { + return ''; + } + + let output = input.trim(); + if ( output.includes( ',' ) ) { + output = output + .split( ',' ) + // finds the first item that is not an empty string. + .find( ( item ) => item.trim() !== '' ) + .trim(); + } + // removes leading and trailing quotes. + output = output.replace( /^["']|["']$/g, '' ); + + // Firefox needs the font name to be wrapped in double quotes meanwhile other browsers don't. + if ( window.navigator.userAgent.toLowerCase().includes( 'firefox' ) ) { + output = `"${ output }"`; + } + return output; +} + +/* + * Loads the font face from a URL and adds it to the browser. + * It also adds it to the iframe document. + */ +export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { + let dataSource; + + if ( typeof source === 'string' ) { + dataSource = `url(${ source })`; + // eslint-disable-next-line no-undef + } else if ( source instanceof File ) { + dataSource = await source.arrayBuffer(); + } else { + return; + } + + const newFont = new window.FontFace( + formatFontFaceName( fontFace.fontFamily ), + dataSource, + { + style: fontFace.fontStyle, + weight: fontFace.fontWeight, + } + ); + + const loadedFace = await newFont.load(); + + if ( addTo === 'document' || addTo === 'all' ) { + document.fonts.add( loadedFace ); + } + + if ( addTo === 'iframe' || addTo === 'all' ) { + const iframeDocument = document.querySelector( + 'iframe[name="editor-canvas"]' + ).contentDocument; + iframeDocument.fonts.add( loadedFace ); + } +} + +/* + * Unloads the font face and remove it from the browser. + * It also removes it from the iframe document. + * + * Note that Font faces that were added to the set using the CSS @font-face rule + * remain connected to the corresponding CSS, and cannot be deleted. + * + * @see https://developer.mozilla.org/en-US/docs/Web/API/FontFaceSet/delete. + */ +export function unloadFontFaceInBrowser( fontFace, removeFrom = 'all' ) { + const unloadFontFace = ( fonts ) => { + fonts.forEach( ( f ) => { + if ( + f.family === formatFontFaceName( fontFace?.fontFamily ) && + f.weight === fontFace?.fontWeight && + f.style === fontFace?.fontStyle + ) { + fonts.delete( f ); + } + } ); + }; + + if ( removeFrom === 'document' || removeFrom === 'all' ) { + unloadFontFace( document.fonts ); + } + + if ( removeFrom === 'iframe' || removeFrom === 'all' ) { + const iframeDocument = document.querySelector( + 'iframe[name="editor-canvas"]' + ).contentDocument; + unloadFontFace( iframeDocument.fonts ); + } +} + +function isUrlEncoded( url ) { + if ( typeof url !== 'string' ) { + return false; + } + return url !== decodeURIComponent( url ); +} + +/* + * Retrieves the display source from a font face src. + * + * @param {string|string[]} fontSrc - The font face src. + * @param {string} baseUrl - The base URL to resolve the src. + * @return {string|undefined} The display source or undefined if the input is invalid. + */ +export function getDisplaySrcFromFontFace( fontSrc, baseUrl ) { + if ( ! fontSrc ) { + return; + } + + let src; + if ( Array.isArray( fontSrc ) ) { + src = fontSrc[ 0 ]; + } else { + src = fontSrc; + } + + if ( ! isUrlEncoded( src ) ) { + src = encodeURI( src ); + } + + // If baseUrl is provided, use it to resolve the src. + if ( src.startsWith( 'file:.' ) ) { + src = baseUrl + '/' + src.replace( 'file:./', '' ); + } + + return src; +} From 1e8ef556d85ee6e3777eb23d74e12bec9321ba94 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 4 Sep 2024 15:36:02 -0300 Subject: [PATCH 02/28] remove code not needed --- .../components/editor-fonts-resolver/utils.js | 38 +------------------ 1 file changed, 1 insertion(+), 37 deletions(-) diff --git a/packages/block-editor/src/components/editor-fonts-resolver/utils.js b/packages/block-editor/src/components/editor-fonts-resolver/utils.js index 23f22a45cc308..305d9eb616365 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/utils.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/utils.js @@ -12,7 +12,7 @@ * formatFontFaceName("'Open Sans', sans-serif") => "Open Sans" * formatFontFaceName(", 'Open Sans', 'Helvetica Neue', sans-serif") => "Open Sans" */ -export function formatFontFaceName( input ) { +function formatFontFaceName( input ) { if ( ! input ) { return ''; } @@ -45,8 +45,6 @@ export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { if ( typeof source === 'string' ) { dataSource = `url(${ source })`; // eslint-disable-next-line no-undef - } else if ( source instanceof File ) { - dataSource = await source.arrayBuffer(); } else { return; } @@ -74,40 +72,6 @@ export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { } } -/* - * Unloads the font face and remove it from the browser. - * It also removes it from the iframe document. - * - * Note that Font faces that were added to the set using the CSS @font-face rule - * remain connected to the corresponding CSS, and cannot be deleted. - * - * @see https://developer.mozilla.org/en-US/docs/Web/API/FontFaceSet/delete. - */ -export function unloadFontFaceInBrowser( fontFace, removeFrom = 'all' ) { - const unloadFontFace = ( fonts ) => { - fonts.forEach( ( f ) => { - if ( - f.family === formatFontFaceName( fontFace?.fontFamily ) && - f.weight === fontFace?.fontWeight && - f.style === fontFace?.fontStyle - ) { - fonts.delete( f ); - } - } ); - }; - - if ( removeFrom === 'document' || removeFrom === 'all' ) { - unloadFontFace( document.fonts ); - } - - if ( removeFrom === 'iframe' || removeFrom === 'all' ) { - const iframeDocument = document.querySelector( - 'iframe[name="editor-canvas"]' - ).contentDocument; - unloadFontFace( iframeDocument.fonts ); - } -} - function isUrlEncoded( url ) { if ( typeof url !== 'string' ) { return false; From 65f37545913412a02bf70d16b8967955d8b1b41d Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Thu, 5 Sep 2024 13:12:50 -0300 Subject: [PATCH 03/28] default value when font families aren't defined --- .../block-editor/src/components/editor-fonts-resolver/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/editor-fonts-resolver/index.js index 17fd238c39a8c..c645e30399e86 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/index.js @@ -22,7 +22,7 @@ function EditorFontsResolver() { }, [] ); //Get the fonts from merged theme.json settings.fontFamilies - const [ fontFamilies ] = useGlobalSetting( 'typography.fontFamilies' ); + const [ fontFamilies = [] ] = useGlobalSetting( 'typography.fontFamilies' ); const fontFaces = useMemo( () => { return Object.values( fontFamilies ) From 77f76386a378bdfe6fbf6cfb295e8f048fd52403 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Mon, 9 Sep 2024 09:40:22 -0300 Subject: [PATCH 04/28] comments formatting Co-authored-by: Sarah Norris <1645628+mikachan@users.noreply.github.com> --- .../src/components/editor-fonts-resolver/index.js | 7 ++++--- .../src/components/editor-fonts-resolver/utils.js | 4 ++-- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/editor-fonts-resolver/index.js index c645e30399e86..435da5affc200 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/index.js @@ -16,12 +16,13 @@ function EditorFontsResolver() { const { currentTheme } = useSelect( ( select ) => { return { currentTheme: - // Using 'core' as string to avoid circular dependency importing from @wordpress/core-data - select( 'core' ).getCurrentTheme(), // eslint-disable-line @wordpress/data-no-store-string-literals + // Disable Reason: Using 'core' as string to avoid circular dependency importing from @wordpress/core-data. + // eslint-disable-next-line @wordpress/data-no-store-string-literals + select( 'core' ).getCurrentTheme(), }; }, [] ); - //Get the fonts from merged theme.json settings.fontFamilies + // Get the fonts from merged theme.json settings.fontFamilies. const [ fontFamilies = [] ] = useGlobalSetting( 'typography.fontFamilies' ); const fontFaces = useMemo( () => { diff --git a/packages/block-editor/src/components/editor-fonts-resolver/utils.js b/packages/block-editor/src/components/editor-fonts-resolver/utils.js index 305d9eb616365..66e914910f0a6 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/utils.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/utils.js @@ -21,11 +21,11 @@ function formatFontFaceName( input ) { if ( output.includes( ',' ) ) { output = output .split( ',' ) - // finds the first item that is not an empty string. + // Finds the first item that is not an empty string. .find( ( item ) => item.trim() !== '' ) .trim(); } - // removes leading and trailing quotes. + // Removes leading and trailing quotes. output = output.replace( /^["']|["']$/g, '' ); // Firefox needs the font name to be wrapped in double quotes meanwhile other browsers don't. From 0bed01d59389a691988c926e12bb6ba552d8002f Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Mon, 9 Sep 2024 10:27:49 -0300 Subject: [PATCH 05/28] use callback for loadFontFaceAsset --- .../components/editor-fonts-resolver/index.js | 46 +++++++++---------- 1 file changed, 23 insertions(+), 23 deletions(-) diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/editor-fonts-resolver/index.js index 435da5affc200..1ba220f47b1af 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/index.js @@ -1,7 +1,7 @@ /** * WordPress dependencies */ -import { useState, useEffect, useMemo } from '@wordpress/element'; +import { useState, useEffect, useMemo, useCallback } from '@wordpress/element'; import { useSelect } from '@wordpress/data'; /** @@ -11,7 +11,7 @@ import { useGlobalSetting } from '../global-styles/hooks'; import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; function EditorFontsResolver() { - const [ loadedFontUrls ] = useState( new Set() ); + const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); const { currentTheme } = useSelect( ( select ) => { return { @@ -33,30 +33,30 @@ function EditorFontsResolver() { .flat(); }, [ fontFamilies ] ); - const loadFontFaceAsset = async ( fontFace ) => { - // If the font doesn't have a src, don't load it. - if ( ! fontFace.src ) { - return; - } - // Get the src of the font. - const src = getDisplaySrcFromFontFace( - fontFace.src, - currentTheme?.stylesheet_uri - ); - // If the font is already loaded, don't load it again. - if ( ! src || loadedFontUrls.has( src ) ) { - return; - } - // Load the font in the browser. - loadFontFaceInBrowser( fontFace, src ); - // Add the font to the loaded fonts list. - loadedFontUrls.add( src ); - }; + const loadFontFaceAsset = useCallback( + async ( fontFace ) => { + if ( ! fontFace.src ) { + return; + } + + const src = getDisplaySrcFromFontFace( + fontFace.src, + currentTheme?.stylesheet_uri + ); + + if ( ! src || loadedFontUrls.has( src ) ) { + return; + } + + loadFontFaceInBrowser( fontFace, src ); + setLoadedFontUrls( ( prevUrls ) => new Set( prevUrls ).add( src ) ); + }, + [ currentTheme, loadedFontUrls ] + ); useEffect( () => { fontFaces.forEach( loadFontFaceAsset ); - // eslint-disable-next-line react-hooks/exhaustive-deps - }, [ fontFaces ] ); + }, [ fontFaces, loadFontFaceAsset ] ); return null; } From 48bfe657541705df1a6961a0c533b9533fd607e4 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Mon, 9 Sep 2024 10:35:14 -0300 Subject: [PATCH 06/28] improve syntax --- .../src/components/editor-fonts-resolver/utils.js | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/packages/block-editor/src/components/editor-fonts-resolver/utils.js b/packages/block-editor/src/components/editor-fonts-resolver/utils.js index 66e914910f0a6..f208d66830d31 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/utils.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/utils.js @@ -40,15 +40,10 @@ function formatFontFaceName( input ) { * It also adds it to the iframe document. */ export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { - let dataSource; - - if ( typeof source === 'string' ) { - dataSource = `url(${ source })`; - // eslint-disable-next-line no-undef - } else { + if ( typeof source !== 'string' ) { return; } - + const dataSource = `url(${ source })`; const newFont = new window.FontFace( formatFontFaceName( fontFace.fontFamily ), dataSource, From c6a32f9d7381e4f93a3c471de7214023f4984edd Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Fri, 13 Sep 2024 16:58:24 -0300 Subject: [PATCH 07/28] Move EditorFontsResolver inside EditorStyles, use fontFamilies data from block-editor store --- .../block-editor/src/components/block-canvas/index.js | 3 --- .../src/components/editor-fonts-resolver/index.js | 10 +++++----- .../block-editor/src/components/editor-styles/index.js | 3 ++- 3 files changed, 7 insertions(+), 9 deletions(-) diff --git a/packages/block-editor/src/components/block-canvas/index.js b/packages/block-editor/src/components/block-canvas/index.js index 03f3a4f2eb5f1..9bad3ca22e95f 100644 --- a/packages/block-editor/src/components/block-canvas/index.js +++ b/packages/block-editor/src/components/block-canvas/index.js @@ -15,7 +15,6 @@ import WritingFlow from '../writing-flow'; import { useMouseMoveTypingReset } from '../observe-typing'; import { useBlockSelectionClearer } from '../block-selection-clearer'; import { useBlockCommands } from '../use-block-commands'; -import EditorFontsResolver from '../editor-fonts-resolver'; // EditorStyles is a memoized component, so avoid passing a new // object reference on each render. @@ -44,7 +43,6 @@ export function ExperimentalBlockCanvas( { __unstableContentRef={ localRef } style={ { height, display: 'flex' } } > - - { children } diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/editor-fonts-resolver/index.js index 1ba220f47b1af..97f490e11c40a 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/editor-fonts-resolver/index.js @@ -7,24 +7,24 @@ import { useSelect } from '@wordpress/data'; /** * Internal dependencies */ -import { useGlobalSetting } from '../global-styles/hooks'; import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; +import { store as editorStore } from '../../store'; function EditorFontsResolver() { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); - const { currentTheme } = useSelect( ( select ) => { + const { currentTheme, fontFamilies = [] } = useSelect( ( select ) => { return { currentTheme: // Disable Reason: Using 'core' as string to avoid circular dependency importing from @wordpress/core-data. // eslint-disable-next-line @wordpress/data-no-store-string-literals select( 'core' ).getCurrentTheme(), + fontFamilies: + select( editorStore ).getSettings()?.__experimentalFeatures + ?.typography?.fontFamilies, }; }, [] ); - // Get the fonts from merged theme.json settings.fontFamilies. - const [ fontFamilies = [] ] = useGlobalSetting( 'typography.fontFamilies' ); - const fontFaces = useMemo( () => { return Object.values( fontFamilies ) .flat() diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index a59ac310bcd30..5108e8fae7b51 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -18,6 +18,7 @@ import { useSelect } from '@wordpress/data'; import transformStyles from '../../utils/transform-styles'; import { store as blockEditorStore } from '../../store'; import { unlock } from '../../lock-unlock'; +import EditorFontsResolver from '../editor-fonts-resolver'; extend( [ namesPlugin, a11yPlugin ] ); @@ -27,7 +28,6 @@ function useDarkThemeBodyClassName( styles, scope ) { if ( ! node ) { return; } - const { ownerDocument } = node; const { defaultView, body } = ownerDocument; const canvas = scope ? ownerDocument.querySelector( scope ) : body; @@ -122,6 +122,7 @@ function EditorStyles( { styles, scope, transformOptions } ) { } } dangerouslySetInnerHTML={ { __html: transformedSvgs } } /> + ); } From 5a9e2d0cc698aec84edf6e3809f8f379a537325c Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Fri, 13 Sep 2024 18:15:54 -0300 Subject: [PATCH 08/28] use a ref to reference the current document --- .../src/components/editor-styles/index.js | 88 +++++++++---------- .../index.js | 8 +- .../utils.js | 15 ++-- 3 files changed, 52 insertions(+), 59 deletions(-) rename packages/block-editor/src/components/{editor-fonts-resolver => use-editor-fonts-resolver}/index.js (91%) rename packages/block-editor/src/components/{editor-fonts-resolver => use-editor-fonts-resolver}/utils.js (86%) diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index 5108e8fae7b51..2a52c0de077a3 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -9,7 +9,7 @@ import a11yPlugin from 'colord/plugins/a11y'; * WordPress dependencies */ import { SVG } from '@wordpress/components'; -import { useCallback, useMemo, memo } from '@wordpress/element'; +import { useMemo, memo, useRef } from '@wordpress/element'; import { useSelect } from '@wordpress/data'; /** @@ -18,53 +18,49 @@ import { useSelect } from '@wordpress/data'; import transformStyles from '../../utils/transform-styles'; import { store as blockEditorStore } from '../../store'; import { unlock } from '../../lock-unlock'; -import EditorFontsResolver from '../editor-fonts-resolver'; +import useEditorFontsResolver from '../use-editor-fonts-resolver'; extend( [ namesPlugin, a11yPlugin ] ); -function useDarkThemeBodyClassName( styles, scope ) { - return useCallback( - ( node ) => { - if ( ! node ) { - return; - } - const { ownerDocument } = node; - const { defaultView, body } = ownerDocument; - const canvas = scope ? ownerDocument.querySelector( scope ) : body; +function useDarkThemeBodyClassName( scope, ref ) { + if ( ! ref.current ) { + return; + } - let backgroundColor; + const { ownerDocument } = ref.current; + const { defaultView, body } = ownerDocument; + const canvas = scope ? ownerDocument.querySelector( scope ) : body; - if ( ! canvas ) { - // The real .editor-styles-wrapper element might not exist in the - // DOM, so calculate the background color by creating a fake - // wrapper. - const tempCanvas = ownerDocument.createElement( 'div' ); - tempCanvas.classList.add( 'editor-styles-wrapper' ); - body.appendChild( tempCanvas ); + let backgroundColor; - backgroundColor = defaultView - ?.getComputedStyle( tempCanvas, null ) - .getPropertyValue( 'background-color' ); + if ( ! canvas ) { + // The real .editor-styles-wrapper element might not exist in the + // DOM, so calculate the background color by creating a fake + // wrapper. + const tempCanvas = ownerDocument.createElement( 'div' ); + tempCanvas.classList.add( 'editor-styles-wrapper' ); + body.appendChild( tempCanvas ); - body.removeChild( tempCanvas ); - } else { - backgroundColor = defaultView - ?.getComputedStyle( canvas, null ) - .getPropertyValue( 'background-color' ); - } - const colordBackgroundColor = colord( backgroundColor ); - // If background is transparent, it should be treated as light color. - if ( - colordBackgroundColor.luminance() > 0.5 || - colordBackgroundColor.alpha() === 0 - ) { - body.classList.remove( 'is-dark-theme' ); - } else { - body.classList.add( 'is-dark-theme' ); - } - }, - [ styles, scope ] - ); + backgroundColor = defaultView + ?.getComputedStyle( tempCanvas, null ) + .getPropertyValue( 'background-color' ); + + body.removeChild( tempCanvas ); + } else { + backgroundColor = defaultView + ?.getComputedStyle( canvas, null ) + .getPropertyValue( 'background-color' ); + } + const colordBackgroundColor = colord( backgroundColor ); + // If background is transparent, it should be treated as light color. + if ( + colordBackgroundColor.luminance() > 0.5 || + colordBackgroundColor.alpha() === 0 + ) { + body.classList.remove( 'is-dark-theme' ); + } else { + body.classList.add( 'is-dark-theme' ); + } } function EditorStyles( { styles, scope, transformOptions } ) { @@ -98,13 +94,16 @@ function EditorStyles( { styles, scope, transformOptions } ) { ]; }, [ styles, overrides, scope, transformOptions ] ); + const styleRef = useRef( null ); + useDarkThemeBodyClassName( scope, styleRef ); + useEditorFontsResolver( styleRef ); + return ( <> { /* Use an empty style element to have a document reference, but this could be any element. */ } - ) ) } @@ -122,7 +121,6 @@ function EditorStyles( { styles, scope, transformOptions } ) { } } dangerouslySetInnerHTML={ { __html: transformedSvgs } } /> - ); } diff --git a/packages/block-editor/src/components/editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js similarity index 91% rename from packages/block-editor/src/components/editor-fonts-resolver/index.js rename to packages/block-editor/src/components/use-editor-fonts-resolver/index.js index 97f490e11c40a..322fd557c129d 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -10,7 +10,7 @@ import { useSelect } from '@wordpress/data'; import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; import { store as editorStore } from '../../store'; -function EditorFontsResolver() { +function useEditorFontsResolver( ref ) { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); const { currentTheme, fontFamilies = [] } = useSelect( ( select ) => { @@ -48,7 +48,7 @@ function EditorFontsResolver() { return; } - loadFontFaceInBrowser( fontFace, src ); + loadFontFaceInBrowser( fontFace, src, ref.current.ownerDocument ); setLoadedFontUrls( ( prevUrls ) => new Set( prevUrls ).add( src ) ); }, [ currentTheme, loadedFontUrls ] @@ -57,8 +57,6 @@ function EditorFontsResolver() { useEffect( () => { fontFaces.forEach( loadFontFaceAsset ); }, [ fontFaces, loadFontFaceAsset ] ); - - return null; } -export default EditorFontsResolver; +export default useEditorFontsResolver; diff --git a/packages/block-editor/src/components/editor-fonts-resolver/utils.js b/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js similarity index 86% rename from packages/block-editor/src/components/editor-fonts-resolver/utils.js rename to packages/block-editor/src/components/use-editor-fonts-resolver/utils.js index f208d66830d31..467eb4ef82a01 100644 --- a/packages/block-editor/src/components/editor-fonts-resolver/utils.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js @@ -39,7 +39,7 @@ function formatFontFaceName( input ) { * Loads the font face from a URL and adds it to the browser. * It also adds it to the iframe document. */ -export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { +export async function loadFontFaceInBrowser( fontFace, source, documentRef ) { if ( typeof source !== 'string' ) { return; } @@ -55,15 +55,12 @@ export async function loadFontFaceInBrowser( fontFace, source, addTo = 'all' ) { const loadedFace = await newFont.load(); - if ( addTo === 'document' || addTo === 'all' ) { - document.fonts.add( loadedFace ); - } + // Add the font to the ref document. + documentRef.fonts.add( loadedFace ); - if ( addTo === 'iframe' || addTo === 'all' ) { - const iframeDocument = document.querySelector( - 'iframe[name="editor-canvas"]' - ).contentDocument; - iframeDocument.fonts.add( loadedFace ); + // Add the font to the window document. + if ( documentRef !== window.document ) { + window.document.fonts.add( loadedFace ); } } From 27b5e6441520f7340af434ca1688e7c5aaa1c381 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Mon, 16 Sep 2024 09:49:30 -0300 Subject: [PATCH 09/28] currentTheme default to empty object --- .../src/components/use-editor-fonts-resolver/index.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js index 322fd557c129d..d6af640274b93 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -13,12 +13,12 @@ import { store as editorStore } from '../../store'; function useEditorFontsResolver( ref ) { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); - const { currentTheme, fontFamilies = [] } = useSelect( ( select ) => { + const { currentTheme = {}, fontFamilies = [] } = useSelect( ( select ) => { return { currentTheme: // Disable Reason: Using 'core' as string to avoid circular dependency importing from @wordpress/core-data. // eslint-disable-next-line @wordpress/data-no-store-string-literals - select( 'core' ).getCurrentTheme(), + select( 'core' )?.getCurrentTheme(), fontFamilies: select( editorStore ).getSettings()?.__experimentalFeatures ?.typography?.fontFamilies, From 463fa952b9cb2170e89398a36625f4195faca6eb Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 18 Sep 2024 09:07:35 -0300 Subject: [PATCH 10/28] revert changes on useDarkThemeBodyClassName, refactor useEditorFontsResolver follwing useDarkThemeBodyClassName pattern --- .../src/components/editor-styles/index.js | 95 ++++++++++--------- .../use-editor-fonts-resolver/index.js | 24 +++-- 2 files changed, 67 insertions(+), 52 deletions(-) diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index 2a52c0de077a3..ca3355596dc5f 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -9,7 +9,7 @@ import a11yPlugin from 'colord/plugins/a11y'; * WordPress dependencies */ import { SVG } from '@wordpress/components'; -import { useMemo, memo, useRef } from '@wordpress/element'; +import { useMemo, memo, useCallback } from '@wordpress/element'; import { useSelect } from '@wordpress/data'; /** @@ -22,45 +22,50 @@ import useEditorFontsResolver from '../use-editor-fonts-resolver'; extend( [ namesPlugin, a11yPlugin ] ); -function useDarkThemeBodyClassName( scope, ref ) { - if ( ! ref.current ) { - return; - } - - const { ownerDocument } = ref.current; - const { defaultView, body } = ownerDocument; - const canvas = scope ? ownerDocument.querySelector( scope ) : body; - - let backgroundColor; - - if ( ! canvas ) { - // The real .editor-styles-wrapper element might not exist in the - // DOM, so calculate the background color by creating a fake - // wrapper. - const tempCanvas = ownerDocument.createElement( 'div' ); - tempCanvas.classList.add( 'editor-styles-wrapper' ); - body.appendChild( tempCanvas ); - - backgroundColor = defaultView - ?.getComputedStyle( tempCanvas, null ) - .getPropertyValue( 'background-color' ); - - body.removeChild( tempCanvas ); - } else { - backgroundColor = defaultView - ?.getComputedStyle( canvas, null ) - .getPropertyValue( 'background-color' ); - } - const colordBackgroundColor = colord( backgroundColor ); - // If background is transparent, it should be treated as light color. - if ( - colordBackgroundColor.luminance() > 0.5 || - colordBackgroundColor.alpha() === 0 - ) { - body.classList.remove( 'is-dark-theme' ); - } else { - body.classList.add( 'is-dark-theme' ); - } +function useDarkThemeBodyClassName( styles, scope ) { + return useCallback( + ( node ) => { + if ( ! node ) { + return; + } + + const { ownerDocument } = node; + const { defaultView, body } = ownerDocument; + const canvas = scope ? ownerDocument.querySelector( scope ) : body; + + let backgroundColor; + + if ( ! canvas ) { + // The real .editor-styles-wrapper element might not exist in the + // DOM, so calculate the background color by creating a fake + // wrapper. + const tempCanvas = ownerDocument.createElement( 'div' ); + tempCanvas.classList.add( 'editor-styles-wrapper' ); + body.appendChild( tempCanvas ); + + backgroundColor = defaultView + ?.getComputedStyle( tempCanvas, null ) + .getPropertyValue( 'background-color' ); + + body.removeChild( tempCanvas ); + } else { + backgroundColor = defaultView + ?.getComputedStyle( canvas, null ) + .getPropertyValue( 'background-color' ); + } + const colordBackgroundColor = colord( backgroundColor ); + // If background is transparent, it should be treated as light color. + if ( + colordBackgroundColor.luminance() > 0.5 || + colordBackgroundColor.alpha() === 0 + ) { + body.classList.remove( 'is-dark-theme' ); + } else { + body.classList.add( 'is-dark-theme' ); + } + }, + [ styles, scope ] + ); } function EditorStyles( { styles, scope, transformOptions } ) { @@ -94,15 +99,15 @@ function EditorStyles( { styles, scope, transformOptions } ) { ]; }, [ styles, overrides, scope, transformOptions ] ); - const styleRef = useRef( null ); - useDarkThemeBodyClassName( scope, styleRef ); - useEditorFontsResolver( styleRef ); - return ( <> { /* Use an empty style element to have a document reference, but this could be any element. */ } - diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js index d6af640274b93..d5cd6ab153590 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -1,7 +1,7 @@ /** * WordPress dependencies */ -import { useState, useEffect, useMemo, useCallback } from '@wordpress/element'; +import { useState, useMemo, useCallback } from '@wordpress/element'; import { useSelect } from '@wordpress/data'; /** @@ -10,7 +10,7 @@ import { useSelect } from '@wordpress/data'; import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; import { store as editorStore } from '../../store'; -function useEditorFontsResolver( ref ) { +function useEditorFontsResolver() { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); const { currentTheme = {}, fontFamilies = [] } = useSelect( ( select ) => { @@ -34,7 +34,7 @@ function useEditorFontsResolver( ref ) { }, [ fontFamilies ] ); const loadFontFaceAsset = useCallback( - async ( fontFace ) => { + async ( fontFace, ownerDocument ) => { if ( ! fontFace.src ) { return; } @@ -48,15 +48,25 @@ function useEditorFontsResolver( ref ) { return; } - loadFontFaceInBrowser( fontFace, src, ref.current.ownerDocument ); + loadFontFaceInBrowser( fontFace, src, ownerDocument ); setLoadedFontUrls( ( prevUrls ) => new Set( prevUrls ).add( src ) ); }, [ currentTheme, loadedFontUrls ] ); - useEffect( () => { - fontFaces.forEach( loadFontFaceAsset ); - }, [ fontFaces, loadFontFaceAsset ] ); + return useCallback( + ( node ) => { + if ( ! node ) { + return; + } + + const { ownerDocument } = node; + fontFaces.forEach( ( fontFace ) => + loadFontFaceAsset( fontFace, ownerDocument ) + ); + }, + [ fontFaces, loadFontFaceAsset ] + ); } export default useEditorFontsResolver; From 9bd3378752eb762cd160251c20f3a35b7e494526 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 18 Sep 2024 09:16:54 -0300 Subject: [PATCH 11/28] revert not needed change --- packages/block-editor/src/components/editor-styles/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index ca3355596dc5f..8af3b493e4496 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -9,7 +9,7 @@ import a11yPlugin from 'colord/plugins/a11y'; * WordPress dependencies */ import { SVG } from '@wordpress/components'; -import { useMemo, memo, useCallback } from '@wordpress/element'; +import { useCallback, useMemo, memo } from '@wordpress/element'; import { useSelect } from '@wordpress/data'; /** From 8ced871bc797e8b4d3819892e01ea7133002938f Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Fri, 20 Sep 2024 10:23:57 -0300 Subject: [PATCH 12/28] try adding currentTheme to the editor settings --- .../src/components/use-editor-fonts-resolver/index.js | 5 ++--- .../src/components/global-styles-renderer/index.js | 11 +++++++++-- 2 files changed, 11 insertions(+), 5 deletions(-) diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js index d5cd6ab153590..a63111614b23e 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -16,9 +16,8 @@ function useEditorFontsResolver() { const { currentTheme = {}, fontFamilies = [] } = useSelect( ( select ) => { return { currentTheme: - // Disable Reason: Using 'core' as string to avoid circular dependency importing from @wordpress/core-data. - // eslint-disable-next-line @wordpress/data-no-store-string-literals - select( 'core' )?.getCurrentTheme(), + select( editorStore ).getSettings()?.__experimentalFeatures + ?.currentTheme, fontFamilies: select( editorStore ).getSettings()?.__experimentalFeatures ?.typography?.fontFamilies, diff --git a/packages/edit-site/src/components/global-styles-renderer/index.js b/packages/edit-site/src/components/global-styles-renderer/index.js index 2e840a7acdc37..0b65627a7bb44 100644 --- a/packages/edit-site/src/components/global-styles-renderer/index.js +++ b/packages/edit-site/src/components/global-styles-renderer/index.js @@ -4,6 +4,7 @@ import { useEffect } from '@wordpress/element'; import { useSelect, useDispatch } from '@wordpress/data'; import { privateApis as blockEditorPrivateApis } from '@wordpress/block-editor'; +import { store as coreDataStore } from '@wordpress/core-data'; /** * Internal dependencies @@ -15,12 +16,18 @@ import { TEMPLATE_POST_TYPE } from '../../utils/constants'; const { useGlobalStylesOutput } = unlock( blockEditorPrivateApis ); function useGlobalStylesRenderer() { - const postType = useSelect( ( select ) => { - return select( editSiteStore ).getEditedPostType(); + const { postType, currentTheme } = useSelect( ( select ) => { + return { + postType: select( editSiteStore ).getEditedPostType(), + currentTheme: select( coreDataStore ).getCurrentTheme(), + }; } ); const [ styles, settings ] = useGlobalStylesOutput( postType !== TEMPLATE_POST_TYPE ); + + settings.currentTheme = currentTheme; + const { getSettings } = useSelect( editSiteStore ); const { updateSettings } = useDispatch( editSiteStore ); From ee98b6ecbefe1d68f6d53d3a52b4550f34f1bade Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Thu, 26 Sep 2024 19:51:43 -0300 Subject: [PATCH 13/28] add theme fonts uris --- ...class-wp-theme-json-resolver-gutenberg.php | 50 +++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index 2231cb0f11538..9c64e09b7e0a3 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -831,6 +831,50 @@ public static function get_style_variations_from_directory( $directory, $scope = return $variations; } + /** + * Resolves relative paths in theme.json typography to theme absolute paths + * and returns them in an array that can be embedded + * as the value of `_link` object in REST API responses. + * + * @since 6.6.0 + * + * @param + * @return array An array of resolved paths. + */ + private static function get_resolved_fonts_theme_uris( $theme_json_data ) { + $resolved_theme_uris = array(); + + if ( !empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { + + $font_families = ( $theme_json_data['settings']['typography']['fontFamilies']['theme'] ?? array() ) + + ( $theme_json_data['settings']['typography']['fontFamilies']['custom'] ?? array() ) + + ( $theme_json_data['settings']['typography']['fontFamilies']['default'] ?? array() ); + + foreach ( $font_families as $font_family ) { + if ( !empty( $font_family['fontFace'] ) ) { + foreach ( $font_family['fontFace'] as $font_face ) { + if ( !empty( $font_face['src'] ) ) { + $sources = is_string( $font_face['src'] ) + ? array( $font_face['src'] ) + : $font_face['src']; + foreach ( $sources as $source ) { + if ( str_starts_with( $source, 'file:' ) ) { + $resolved_theme_uris[] = array( + 'name' => $source, + 'href' => sanitize_url( get_theme_file_uri( str_replace( 'file:./', '', $source ) ) ), + 'target' => "typography.fontFamilies.{$font_family['slug']}.fontFace.src", + ); + } + } + } + } + } + } + } + + return $resolved_theme_uris; + } + /** * Resolves relative paths in theme.json styles to theme absolute paths @@ -852,6 +896,12 @@ public static function get_resolved_theme_uris( $theme_json ) { $theme_json_data = $theme_json->get_raw_data(); + // Add font URIs. + $resolved_theme_uris = array_merge( + $resolved_theme_uris, + static::get_resolved_fonts_theme_uris( $theme_json_data ) + ); + // Using the same file convention when registering web fonts. See: WP_Font_Face_Resolver:: to_theme_file_uri. $placeholder = 'file:./'; From 146198902f9e68f23681157bcafe4497143f6cdb Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Fri, 27 Sep 2024 14:00:50 -0300 Subject: [PATCH 14/28] use _links in the font resolver --- .../use-editor-fonts-resolver/index.js | 29 ++++++++------ .../use-editor-fonts-resolver/utils.js | 38 ------------------- 2 files changed, 17 insertions(+), 50 deletions(-) diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js index a63111614b23e..4539464d52a7d 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -7,23 +7,31 @@ import { useSelect } from '@wordpress/data'; /** * Internal dependencies */ -import { getDisplaySrcFromFontFace, loadFontFaceInBrowser } from './utils'; +import { loadFontFaceInBrowser } from './utils'; import { store as editorStore } from '../../store'; +import { globalStylesLinksDataKey } from '../../store/private-keys'; + +function resolveThemeFontFaceSrc( src, _links ) { + const firstSrc = Array.isArray( src ) ? src[ 0 ] : src; + return _links.find( ( link ) => link.name === firstSrc )?.href || firstSrc; +} function useEditorFontsResolver() { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); - const { currentTheme = {}, fontFamilies = [] } = useSelect( ( select ) => { + const { _links = [], fontFamilies = [] } = useSelect( ( select ) => { + const { getSettings } = select( editorStore ); + const _settings = getSettings(); return { - currentTheme: - select( editorStore ).getSettings()?.__experimentalFeatures - ?.currentTheme, + _links: _settings[ globalStylesLinksDataKey ]?.[ 'wp:theme-file' ], fontFamilies: - select( editorStore ).getSettings()?.__experimentalFeatures - ?.typography?.fontFamilies, + _settings?.__experimentalFeatures?.typography?.fontFamilies, }; }, [] ); + // eslint-disable-next-line no-console + console.log( 'fontFamilies', fontFamilies, 'links', _links ); + const fontFaces = useMemo( () => { return Object.values( fontFamilies ) .flat() @@ -38,10 +46,7 @@ function useEditorFontsResolver() { return; } - const src = getDisplaySrcFromFontFace( - fontFace.src, - currentTheme?.stylesheet_uri - ); + const src = resolveThemeFontFaceSrc( fontFace.src, _links ); if ( ! src || loadedFontUrls.has( src ) ) { return; @@ -50,7 +55,7 @@ function useEditorFontsResolver() { loadFontFaceInBrowser( fontFace, src, ownerDocument ); setLoadedFontUrls( ( prevUrls ) => new Set( prevUrls ).add( src ) ); }, - [ currentTheme, loadedFontUrls ] + [ loadedFontUrls, _links ] ); return useCallback( diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js b/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js index 467eb4ef82a01..4bd2f5db26aa0 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js @@ -63,41 +63,3 @@ export async function loadFontFaceInBrowser( fontFace, source, documentRef ) { window.document.fonts.add( loadedFace ); } } - -function isUrlEncoded( url ) { - if ( typeof url !== 'string' ) { - return false; - } - return url !== decodeURIComponent( url ); -} - -/* - * Retrieves the display source from a font face src. - * - * @param {string|string[]} fontSrc - The font face src. - * @param {string} baseUrl - The base URL to resolve the src. - * @return {string|undefined} The display source or undefined if the input is invalid. - */ -export function getDisplaySrcFromFontFace( fontSrc, baseUrl ) { - if ( ! fontSrc ) { - return; - } - - let src; - if ( Array.isArray( fontSrc ) ) { - src = fontSrc[ 0 ]; - } else { - src = fontSrc; - } - - if ( ! isUrlEncoded( src ) ) { - src = encodeURI( src ); - } - - // If baseUrl is provided, use it to resolve the src. - if ( src.startsWith( 'file:.' ) ) { - src = baseUrl + '/' + src.replace( 'file:./', '' ); - } - - return src; -} From 7cb4c01f6392f2161462752cb56c073b3c0b9cd0 Mon Sep 17 00:00:00 2001 From: Dave Smith Date: Mon, 7 Oct 2024 11:29:14 +0100 Subject: [PATCH 15/28] Fix PHP linting --- lib/class-wp-theme-json-resolver-gutenberg.php | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index 9c64e09b7e0a3..8d520256a2da2 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -838,30 +838,30 @@ public static function get_style_variations_from_directory( $directory, $scope = * * @since 6.6.0 * - * @param + * @param * @return array An array of resolved paths. */ private static function get_resolved_fonts_theme_uris( $theme_json_data ) { $resolved_theme_uris = array(); - if ( !empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { + if ( ! empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { $font_families = ( $theme_json_data['settings']['typography']['fontFamilies']['theme'] ?? array() ) + ( $theme_json_data['settings']['typography']['fontFamilies']['custom'] ?? array() ) + ( $theme_json_data['settings']['typography']['fontFamilies']['default'] ?? array() ); foreach ( $font_families as $font_family ) { - if ( !empty( $font_family['fontFace'] ) ) { + if ( ! empty( $font_family['fontFace'] ) ) { foreach ( $font_family['fontFace'] as $font_face ) { - if ( !empty( $font_face['src'] ) ) { - $sources = is_string( $font_face['src'] ) + if ( ! empty( $font_face['src'] ) ) { + $sources = is_string( $font_face['src'] ) ? array( $font_face['src'] ) : $font_face['src']; foreach ( $sources as $source ) { if ( str_starts_with( $source, 'file:' ) ) { $resolved_theme_uris[] = array( - 'name' => $source, - 'href' => sanitize_url( get_theme_file_uri( str_replace( 'file:./', '', $source ) ) ), + 'name' => $source, + 'href' => sanitize_url( get_theme_file_uri( str_replace( 'file:./', '', $source ) ) ), 'target' => "typography.fontFamilies.{$font_family['slug']}.fontFace.src", ); } From 20a00e3cd1eab0ef4ec3c97a63d50c802db8840b Mon Sep 17 00:00:00 2001 From: Dave Smith Date: Mon, 7 Oct 2024 11:32:58 +0100 Subject: [PATCH 16/28] Use correct alias --- .../src/components/use-editor-fonts-resolver/index.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js index 4539464d52a7d..c240e76b29309 100644 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js @@ -8,7 +8,7 @@ import { useSelect } from '@wordpress/data'; * Internal dependencies */ import { loadFontFaceInBrowser } from './utils'; -import { store as editorStore } from '../../store'; +import { store as blockEditorStore } from '../../store'; import { globalStylesLinksDataKey } from '../../store/private-keys'; function resolveThemeFontFaceSrc( src, _links ) { @@ -20,7 +20,7 @@ function useEditorFontsResolver() { const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); const { _links = [], fontFamilies = [] } = useSelect( ( select ) => { - const { getSettings } = select( editorStore ); + const { getSettings } = select( blockEditorStore ); const _settings = getSettings(); return { _links: _settings[ globalStylesLinksDataKey ]?.[ 'wp:theme-file' ], From f2ad8e7bf1ddb97b84908524b70b9272b15a4826 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:11:42 -0300 Subject: [PATCH 17/28] load the fonts as styles --- .../src/components/editor-styles/index.js | 5 +- .../global-styles/use-global-styles-output.js | 71 ++++++++++++++++- .../use-editor-fonts-resolver/index.js | 76 ------------------- .../use-editor-fonts-resolver/utils.js | 65 ---------------- .../variations/variations-typography.js | 2 +- 5 files changed, 71 insertions(+), 148 deletions(-) delete mode 100644 packages/block-editor/src/components/use-editor-fonts-resolver/index.js delete mode 100644 packages/block-editor/src/components/use-editor-fonts-resolver/utils.js diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index 8af3b493e4496..5d6bd6a7d3c58 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -18,7 +18,6 @@ import { useSelect } from '@wordpress/data'; import transformStyles from '../../utils/transform-styles'; import { store as blockEditorStore } from '../../store'; import { unlock } from '../../lock-unlock'; -import useEditorFontsResolver from '../use-editor-fonts-resolver'; extend( [ namesPlugin, a11yPlugin ] ); @@ -64,7 +63,7 @@ function useDarkThemeBodyClassName( styles, scope ) { body.classList.add( 'is-dark-theme' ); } }, - [ styles, scope ] + [ scope ] ); } @@ -107,8 +106,6 @@ function EditorStyles( { styles, scope, transformOptions } ) { ref={ useDarkThemeBodyClassName( transformedStyles, scope ) } /> - ) ) } diff --git a/packages/block-editor/src/components/global-styles/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/use-global-styles-output.js index cd4ad0cea50e0..6ccb20993d3ba 100644 --- a/packages/block-editor/src/components/global-styles/use-global-styles-output.js +++ b/packages/block-editor/src/components/global-styles/use-global-styles-output.js @@ -25,6 +25,7 @@ import { appendToSelector, getBlockStyleVariationSelector, getResolvedValue, + getResolvedThemeFilePath, } from './utils'; import { getBlockCSSSelector } from './get-block-css-selector'; import { getTypographyFontSizeValue } from './typography-utils'; @@ -884,6 +885,72 @@ export const toCustomProperties = ( tree, blockSelectors ) => { return ruleset; }; +const getFontFaceDeclarations = ( tree ) => { + const fonts = tree?.settings?.typography?.fontFamilies; + let ruleset = ''; + + if ( ! fonts ) { + return ruleset; + } + + const themeFileURIs = tree?._links[ 'wp:theme-file' ] ?? []; + + // Iterate over main origins (theme, custom, default) + for ( const origin in fonts ) { + if ( Array.isArray( fonts[ origin ] ) ) { + fonts[ origin ].forEach( ( font ) => { + if ( font.fontFace ) { + font.fontFace.forEach( ( face ) => { + ruleset += '@font-face {\n'; + + // Optional properties + for ( const [ key, value ] of Object.entries( face ) ) { + if ( key !== 'src' ) { + ruleset += ` ${ kebabCase( + key + ) }: ${ value };\n`; + } + } + + if ( face.src ) { + const srcs = ( + face.src && Array.isArray( face.src ) + ? face.src + : [ face.src ] + ).map( ( src ) => { + if ( src.startsWith( 'file:' ) ) { + // Convert file path to URL and assume format based on extension + const resolvedSrc = + getResolvedThemeFilePath( + src, + themeFileURIs + ); + const format = resolvedSrc + .split( '.' ) + .pop(); + return `url('${ resolvedSrc.replace( + 'file:', + '' + ) }') format('${ format }')`; + } + // Assume it's already a URL and format is provided + return `url('${ src }')`; + } ); + ruleset += ` src: ${ srcs.join( + ',\n ' + ) };\n`; + } + + ruleset += '}\n\n'; + } ); + } + } ); + } + } + + return ruleset; +}; + export const toStyles = ( tree, blockSelectors, @@ -913,6 +980,8 @@ export const toStyles = ( let ruleset = ''; + ruleset += getFontFaceDeclarations( tree ); + if ( options.presets && ( contentSize || wideSize ) ) { ruleset += `${ ROOT_CSS_PROPERTIES_SELECTOR } {`; ruleset = contentSize @@ -1399,7 +1468,6 @@ export function useGlobalStylesOutputWithConfig( } ); const { getBlockStyles } = useSelect( blocksStore ); - return useMemo( () => { if ( ! mergedConfig?.styles || ! mergedConfig?.settings ) { return []; @@ -1415,7 +1483,6 @@ export function useGlobalStylesOutputWithConfig( updatedConfig, blockSelectors ); - const globalStyles = toStyles( updatedConfig, blockSelectors, diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js b/packages/block-editor/src/components/use-editor-fonts-resolver/index.js deleted file mode 100644 index c240e76b29309..0000000000000 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/index.js +++ /dev/null @@ -1,76 +0,0 @@ -/** - * WordPress dependencies - */ -import { useState, useMemo, useCallback } from '@wordpress/element'; -import { useSelect } from '@wordpress/data'; - -/** - * Internal dependencies - */ -import { loadFontFaceInBrowser } from './utils'; -import { store as blockEditorStore } from '../../store'; -import { globalStylesLinksDataKey } from '../../store/private-keys'; - -function resolveThemeFontFaceSrc( src, _links ) { - const firstSrc = Array.isArray( src ) ? src[ 0 ] : src; - return _links.find( ( link ) => link.name === firstSrc )?.href || firstSrc; -} - -function useEditorFontsResolver() { - const [ loadedFontUrls, setLoadedFontUrls ] = useState( new Set() ); - - const { _links = [], fontFamilies = [] } = useSelect( ( select ) => { - const { getSettings } = select( blockEditorStore ); - const _settings = getSettings(); - return { - _links: _settings[ globalStylesLinksDataKey ]?.[ 'wp:theme-file' ], - fontFamilies: - _settings?.__experimentalFeatures?.typography?.fontFamilies, - }; - }, [] ); - - // eslint-disable-next-line no-console - console.log( 'fontFamilies', fontFamilies, 'links', _links ); - - const fontFaces = useMemo( () => { - return Object.values( fontFamilies ) - .flat() - .map( ( family ) => family.fontFace ) - .filter( Boolean ) - .flat(); - }, [ fontFamilies ] ); - - const loadFontFaceAsset = useCallback( - async ( fontFace, ownerDocument ) => { - if ( ! fontFace.src ) { - return; - } - - const src = resolveThemeFontFaceSrc( fontFace.src, _links ); - - if ( ! src || loadedFontUrls.has( src ) ) { - return; - } - - loadFontFaceInBrowser( fontFace, src, ownerDocument ); - setLoadedFontUrls( ( prevUrls ) => new Set( prevUrls ).add( src ) ); - }, - [ loadedFontUrls, _links ] - ); - - return useCallback( - ( node ) => { - if ( ! node ) { - return; - } - - const { ownerDocument } = node; - fontFaces.forEach( ( fontFace ) => - loadFontFaceAsset( fontFace, ownerDocument ) - ); - }, - [ fontFaces, loadFontFaceAsset ] - ); -} - -export default useEditorFontsResolver; diff --git a/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js b/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js deleted file mode 100644 index 4bd2f5db26aa0..0000000000000 --- a/packages/block-editor/src/components/use-editor-fonts-resolver/utils.js +++ /dev/null @@ -1,65 +0,0 @@ -/* - * Format the font face name to use in the font-family property of a font face. - * - * The input can be a string with the font face name or a string with multiple font face names separated by commas. - * It removes the leading and trailing quotes from the font face name. - * - * @param {string} input - The font face name. - * @return {string} The formatted font face name. - * - * Example: - * formatFontFaceName("Open Sans") => "Open Sans" - * formatFontFaceName("'Open Sans', sans-serif") => "Open Sans" - * formatFontFaceName(", 'Open Sans', 'Helvetica Neue', sans-serif") => "Open Sans" - */ -function formatFontFaceName( input ) { - if ( ! input ) { - return ''; - } - - let output = input.trim(); - if ( output.includes( ',' ) ) { - output = output - .split( ',' ) - // Finds the first item that is not an empty string. - .find( ( item ) => item.trim() !== '' ) - .trim(); - } - // Removes leading and trailing quotes. - output = output.replace( /^["']|["']$/g, '' ); - - // Firefox needs the font name to be wrapped in double quotes meanwhile other browsers don't. - if ( window.navigator.userAgent.toLowerCase().includes( 'firefox' ) ) { - output = `"${ output }"`; - } - return output; -} - -/* - * Loads the font face from a URL and adds it to the browser. - * It also adds it to the iframe document. - */ -export async function loadFontFaceInBrowser( fontFace, source, documentRef ) { - if ( typeof source !== 'string' ) { - return; - } - const dataSource = `url(${ source })`; - const newFont = new window.FontFace( - formatFontFaceName( fontFace.fontFamily ), - dataSource, - { - style: fontFace.fontStyle, - weight: fontFace.fontWeight, - } - ); - - const loadedFace = await newFont.load(); - - // Add the font to the ref document. - documentRef.fonts.add( loadedFace ); - - // Add the font to the window document. - if ( documentRef !== window.document ) { - window.document.fonts.add( loadedFace ); - } -} diff --git a/packages/edit-site/src/components/global-styles/variations/variations-typography.js b/packages/edit-site/src/components/global-styles/variations/variations-typography.js index 65e84a9d965ff..957d3bac97e9f 100644 --- a/packages/edit-site/src/components/global-styles/variations/variations-typography.js +++ b/packages/edit-site/src/components/global-styles/variations/variations-typography.js @@ -15,7 +15,7 @@ import Subtitle from '../subtitle'; import Variation from './variation'; export default function TypographyVariations( { title, gap = 2 } ) { - const propertiesToFilter = [ 'typography' ]; + const propertiesToFilter = [ 'typography', '_links' ]; const typographyVariations = useCurrentMergeThemeStyleVariationsWithUserConfig( propertiesToFilter ); From 5a3b172dd31cd5145ad81ccdca38983ac913f35a Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:23:09 -0300 Subject: [PATCH 18/28] optional chain --- .../src/components/global-styles/use-global-styles-output.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/block-editor/src/components/global-styles/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/use-global-styles-output.js index 6ccb20993d3ba..66e16652a5259 100644 --- a/packages/block-editor/src/components/global-styles/use-global-styles-output.js +++ b/packages/block-editor/src/components/global-styles/use-global-styles-output.js @@ -886,14 +886,14 @@ export const toCustomProperties = ( tree, blockSelectors ) => { }; const getFontFaceDeclarations = ( tree ) => { - const fonts = tree?.settings?.typography?.fontFamilies; + const fonts = tree?.settings?.typography?.fontFamilies ?? []; let ruleset = ''; if ( ! fonts ) { return ruleset; } - const themeFileURIs = tree?._links[ 'wp:theme-file' ] ?? []; + const themeFileURIs = tree?._links?.[ 'wp:theme-file' ] ?? []; // Iterate over main origins (theme, custom, default) for ( const origin in fonts ) { From db89bf6bdebd670c2bfd7089f2750e3ce8d069f1 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:23:44 -0300 Subject: [PATCH 19/28] remove default not needed --- .../src/components/global-styles/use-global-styles-output.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/block-editor/src/components/global-styles/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/use-global-styles-output.js index 66e16652a5259..6f58bc7baffb7 100644 --- a/packages/block-editor/src/components/global-styles/use-global-styles-output.js +++ b/packages/block-editor/src/components/global-styles/use-global-styles-output.js @@ -886,7 +886,7 @@ export const toCustomProperties = ( tree, blockSelectors ) => { }; const getFontFaceDeclarations = ( tree ) => { - const fonts = tree?.settings?.typography?.fontFamilies ?? []; + const fonts = tree?.settings?.typography?.fontFamilies; let ruleset = ''; if ( ! fonts ) { From 32868cdb850db0c1ea2181488f43b2990686b704 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:36:17 -0300 Subject: [PATCH 20/28] use array_merge instead of + --- lib/class-wp-theme-json-resolver-gutenberg.php | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index 8d520256a2da2..f351d5a2ca10f 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -846,9 +846,11 @@ private static function get_resolved_fonts_theme_uris( $theme_json_data ) { if ( ! empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { - $font_families = ( $theme_json_data['settings']['typography']['fontFamilies']['theme'] ?? array() ) - + ( $theme_json_data['settings']['typography']['fontFamilies']['custom'] ?? array() ) - + ( $theme_json_data['settings']['typography']['fontFamilies']['default'] ?? array() ); + $font_families = array_merge( + $theme_json_data['settings']['typography']['fontFamilies']['theme'] ?? array(), + $theme_json_data['settings']['typography']['fontFamilies']['custom'] ?? array(), + $theme_json_data['settings']['typography']['fontFamilies']['default'] ?? array() + ); foreach ( $font_families as $font_family ) { if ( ! empty( $font_family['fontFace'] ) ) { From efd95620151a7dba754fb1d1d4407be34ecbdabb Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:48:16 -0300 Subject: [PATCH 21/28] get the theme font uris inside get_resolved_theme_uris function --- ...class-wp-theme-json-resolver-gutenberg.php | 57 ++++++------------- 1 file changed, 17 insertions(+), 40 deletions(-) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index f351d5a2ca10f..b3a067316fd73 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -832,26 +832,36 @@ public static function get_style_variations_from_directory( $directory, $scope = } /** - * Resolves relative paths in theme.json typography to theme absolute paths + * Resolves relative paths in theme.json styles to theme absolute paths * and returns them in an array that can be embedded * as the value of `_link` object in REST API responses. * * @since 6.6.0 + * @since 6.7.0 Added support for resolving block styles. + * @since 6.7.0 Added support for resolving font faces URIs. * - * @param + * @param WP_Theme_JSON_Gutenberg $theme_json A theme json instance. * @return array An array of resolved paths. */ - private static function get_resolved_fonts_theme_uris( $theme_json_data ) { + public static function get_resolved_theme_uris( $theme_json ) { $resolved_theme_uris = array(); - if ( ! empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { + if ( ! $theme_json instanceof WP_Theme_JSON_Gutenberg ) { + return $resolved_theme_uris; + } + + $theme_json_data = $theme_json->get_raw_data(); + // Using the same file convention when registering web fonts. See: WP_Font_Face_Resolver:: to_theme_file_uri. + $placeholder = 'file:./'; + + // Add font URIs. + if ( ! empty( $theme_json_data['settings']['typography']['fontFamilies'] ) ) { $font_families = array_merge( $theme_json_data['settings']['typography']['fontFamilies']['theme'] ?? array(), $theme_json_data['settings']['typography']['fontFamilies']['custom'] ?? array(), $theme_json_data['settings']['typography']['fontFamilies']['default'] ?? array() ); - foreach ( $font_families as $font_family ) { if ( ! empty( $font_family['fontFace'] ) ) { foreach ( $font_family['fontFace'] as $font_face ) { @@ -860,10 +870,10 @@ private static function get_resolved_fonts_theme_uris( $theme_json_data ) { ? array( $font_face['src'] ) : $font_face['src']; foreach ( $sources as $source ) { - if ( str_starts_with( $source, 'file:' ) ) { + if ( str_starts_with( $source, $placeholder ) ) { $resolved_theme_uris[] = array( 'name' => $source, - 'href' => sanitize_url( get_theme_file_uri( str_replace( 'file:./', '', $source ) ) ), + 'href' => sanitize_url( get_theme_file_uri( str_replace( $placeholder, '', $source ) ) ), 'target' => "typography.fontFamilies.{$font_family['slug']}.fontFace.src", ); } @@ -874,39 +884,6 @@ private static function get_resolved_fonts_theme_uris( $theme_json_data ) { } } - return $resolved_theme_uris; - } - - - /** - * Resolves relative paths in theme.json styles to theme absolute paths - * and returns them in an array that can be embedded - * as the value of `_link` object in REST API responses. - * - * @since 6.6.0 - * @since 6.7.0 Added support for resolving block styles. - * - * @param WP_Theme_JSON_Gutenberg $theme_json A theme json instance. - * @return array An array of resolved paths. - */ - public static function get_resolved_theme_uris( $theme_json ) { - $resolved_theme_uris = array(); - - if ( ! $theme_json instanceof WP_Theme_JSON_Gutenberg ) { - return $resolved_theme_uris; - } - - $theme_json_data = $theme_json->get_raw_data(); - - // Add font URIs. - $resolved_theme_uris = array_merge( - $resolved_theme_uris, - static::get_resolved_fonts_theme_uris( $theme_json_data ) - ); - - // Using the same file convention when registering web fonts. See: WP_Font_Face_Resolver:: to_theme_file_uri. - $placeholder = 'file:./'; - // Top level styles. $background_image_url = $theme_json_data['styles']['background']['backgroundImage']['url'] ?? null; if ( From e85076aff173b25999fa7edfeabe2c56df1d1640 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:55:09 -0300 Subject: [PATCH 22/28] remove unwanted changes --- packages/block-editor/src/components/editor-styles/index.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/packages/block-editor/src/components/editor-styles/index.js b/packages/block-editor/src/components/editor-styles/index.js index 5d6bd6a7d3c58..a59ac310bcd30 100644 --- a/packages/block-editor/src/components/editor-styles/index.js +++ b/packages/block-editor/src/components/editor-styles/index.js @@ -63,7 +63,7 @@ function useDarkThemeBodyClassName( styles, scope ) { body.classList.add( 'is-dark-theme' ); } }, - [ scope ] + [ styles, scope ] ); } @@ -105,7 +105,6 @@ function EditorStyles( { styles, scope, transformOptions } ) { ) ) } From a1cdf2cee4300200d0d4f2fcb23b9fc964cc4808 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 12:58:14 -0300 Subject: [PATCH 23/28] remove unwwanted changes --- .../src/components/global-styles-renderer/index.js | 11 ++--------- 1 file changed, 2 insertions(+), 9 deletions(-) diff --git a/packages/edit-site/src/components/global-styles-renderer/index.js b/packages/edit-site/src/components/global-styles-renderer/index.js index 0b65627a7bb44..2e840a7acdc37 100644 --- a/packages/edit-site/src/components/global-styles-renderer/index.js +++ b/packages/edit-site/src/components/global-styles-renderer/index.js @@ -4,7 +4,6 @@ import { useEffect } from '@wordpress/element'; import { useSelect, useDispatch } from '@wordpress/data'; import { privateApis as blockEditorPrivateApis } from '@wordpress/block-editor'; -import { store as coreDataStore } from '@wordpress/core-data'; /** * Internal dependencies @@ -16,18 +15,12 @@ import { TEMPLATE_POST_TYPE } from '../../utils/constants'; const { useGlobalStylesOutput } = unlock( blockEditorPrivateApis ); function useGlobalStylesRenderer() { - const { postType, currentTheme } = useSelect( ( select ) => { - return { - postType: select( editSiteStore ).getEditedPostType(), - currentTheme: select( coreDataStore ).getCurrentTheme(), - }; + const postType = useSelect( ( select ) => { + return select( editSiteStore ).getEditedPostType(); } ); const [ styles, settings ] = useGlobalStylesOutput( postType !== TEMPLATE_POST_TYPE ); - - settings.currentTheme = currentTheme; - const { getSettings } = useSelect( editSiteStore ); const { updateSettings } = useDispatch( editSiteStore ); From 2250a0c2c84f4013575e5fb8710c509067731536 Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 13:02:03 -0300 Subject: [PATCH 24/28] lint --- lib/class-wp-theme-json-resolver-gutenberg.php | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index b3a067316fd73..1a5dc3926f536 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -831,6 +831,7 @@ public static function get_style_variations_from_directory( $directory, $scope = return $variations; } + /** * Resolves relative paths in theme.json styles to theme absolute paths * and returns them in an array that can be embedded From 59ce5a4067af23eb94c85413156e2aab0e7b952c Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Wed, 9 Oct 2024 16:11:40 -0300 Subject: [PATCH 25/28] lint --- lib/class-wp-theme-json-resolver-gutenberg.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index 1a5dc3926f536..5b440ad0cf121 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -831,7 +831,7 @@ public static function get_style_variations_from_directory( $directory, $scope = return $variations; } - + /** * Resolves relative paths in theme.json styles to theme absolute paths * and returns them in an array that can be embedded From d94e2f2ef8bca2ead0d5c4a2990206809b74612e Mon Sep 17 00:00:00 2001 From: ramon Date: Fri, 11 Oct 2024 12:25:12 +1100 Subject: [PATCH 26/28] Adding a unit test to cover the font file resolution in WP_Theme_JSON_Resolver_Gutenberg::get_resolved_theme_uris --- phpunit/class-wp-theme-json-resolver-test.php | 50 ++++++++++++++++++- 1 file changed, 48 insertions(+), 2 deletions(-) diff --git a/phpunit/class-wp-theme-json-resolver-test.php b/phpunit/class-wp-theme-json-resolver-test.php index d2339f2496290..164d80edc8369 100644 --- a/phpunit/class-wp-theme-json-resolver-test.php +++ b/phpunit/class-wp-theme-json-resolver-test.php @@ -1286,8 +1286,44 @@ public function test_resolve_theme_file_uris() { public function test_get_resolved_theme_uris() { $theme_json = new WP_Theme_JSON_Gutenberg( array( - 'version' => WP_Theme_JSON_Gutenberg::LATEST_SCHEMA, - 'styles' => array( + 'version' => WP_Theme_JSON_Gutenberg::LATEST_SCHEMA, + 'settings' => array( + 'typography' => array( + 'fontFamilies' => array( + array( + 'fontFace' => array( + array( + 'fontFamily' => 'Tocco', + 'fontStyle' => 'normal', + 'fontWeight' => '400', + 'src' => array( + 'file:./example/fonts/tocco/tocco-400-normal.woff2', + ), + ), + ), + 'fontFamily' => 'Tocco, system-ui', + 'name' => 'Tocco', + 'slug' => 'secondary', + ), + array( + 'fontFace' => array( + array( + 'fontFamily' => '"Strozzapreti"', + 'fontStyle' => 'normal', + 'fontWeight' => '400', + 'src' => array( + 'file:./example/fonts/strozzapreti/strozzapreti-400-normal.woff2', + ), + ), + ), + 'fontFamily' => '"Strozzapreti", cursive', + 'name' => 'Strozzapreti', + 'slug' => 'primary', + ), + ), + ), + ), + 'styles' => array( 'background' => array( 'backgroundImage' => array( 'url' => 'file:./example/img/image.png', @@ -1314,6 +1350,16 @@ public function test_get_resolved_theme_uris() { ); $expected_data = array( + array( + 'name' => 'file:./example/fonts/tocco/tocco-400-normal.woff2', + 'href' => 'https://example.org/wp-content/themes/example-theme/example/fonts/tocco/tocco-400-normal.woff2', + 'target' => 'typography.fontFamilies.secondary.fontFace.src', + ), + array( + 'name' => 'file:./example/fonts/strozzapreti/strozzapreti-400-normal.woff2', + 'href' => 'https://example.org/wp-content/themes/example-theme/example/fonts/strozzapreti/strozzapreti-400-normal.woff2', + 'target' => 'typography.fontFamilies.primary.fontFace.src', + ), array( 'name' => 'file:./example/img/image.png', 'href' => 'https://example.org/wp-content/themes/example-theme/example/img/image.png', From 607740aac384578356f07cf9bfb19989a738739b Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Mon, 14 Oct 2024 12:00:50 -0300 Subject: [PATCH 27/28] update comment Co-authored-by: Ramon --- lib/class-wp-theme-json-resolver-gutenberg.php | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/lib/class-wp-theme-json-resolver-gutenberg.php b/lib/class-wp-theme-json-resolver-gutenberg.php index 5b440ad0cf121..211108532c69b 100644 --- a/lib/class-wp-theme-json-resolver-gutenberg.php +++ b/lib/class-wp-theme-json-resolver-gutenberg.php @@ -838,8 +838,7 @@ public static function get_style_variations_from_directory( $directory, $scope = * as the value of `_link` object in REST API responses. * * @since 6.6.0 - * @since 6.7.0 Added support for resolving block styles. - * @since 6.7.0 Added support for resolving font faces URIs. + * @since 6.7.0 Added support for resolving block style and font face URIs. * * @param WP_Theme_JSON_Gutenberg $theme_json A theme json instance. * @return array An array of resolved paths. From b69ae26276c3db6519fc790d1415500373363bfc Mon Sep 17 00:00:00 2001 From: Matias Benedetto Date: Tue, 15 Oct 2024 13:57:05 -0300 Subject: [PATCH 28/28] add function comment --- .../global-styles/use-global-styles-output.js | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/packages/block-editor/src/components/global-styles/use-global-styles-output.js b/packages/block-editor/src/components/global-styles/use-global-styles-output.js index 6f58bc7baffb7..b37debe38cde4 100644 --- a/packages/block-editor/src/components/global-styles/use-global-styles-output.js +++ b/packages/block-editor/src/components/global-styles/use-global-styles-output.js @@ -885,6 +885,25 @@ export const toCustomProperties = ( tree, blockSelectors ) => { return ruleset; }; +/** + * Generates CSS @font-face declarations based on font settings in the theme.json tree. + * + * This function processes font families defined in the theme's typography settings, + * creating @font-face rules for each font face. It handles both file-based and URL-based + * font sources, resolving file paths to URLs when necessary. + * + * @param {Object} tree - The theme.json tree containing typography settings. + * @return {string} A string of CSS @font-face rules for the defined fonts. + * + * @example + * // Example output: + * // @font-face { + * // font-family: "Open Sans"; + * // font-style: normal; + * // font-weight: 400; + * // src: url('https://example.com/wp-content/themes/example/assets/fonts/open-sans-regular.woff2') format('woff2'); + * // } + */ const getFontFaceDeclarations = ( tree ) => { const fonts = tree?.settings?.typography?.fontFamilies; let ruleset = '';