| <?php |
| |
| |
| |
| |
| |
| |
| |
|
|
| |
| |
| |
| |
| |
| class WP_Font_Face { |
|
|
| |
| |
| |
| |
| |
| |
| |
| private $font_face_property_defaults = array( |
| 'font-family' => '', |
| 'font-style' => 'normal', |
| 'font-weight' => '400', |
| 'font-display' => 'fallback', |
| ); |
|
|
| |
| |
| |
| |
| |
| |
| |
| private $valid_font_face_properties = array( |
| 'ascent-override', |
| 'descent-override', |
| 'font-display', |
| 'font-family', |
| 'font-stretch', |
| 'font-style', |
| 'font-weight', |
| 'font-variant', |
| 'font-feature-settings', |
| 'font-variation-settings', |
| 'line-gap-override', |
| 'size-adjust', |
| 'src', |
| 'unicode-range', |
| ); |
|
|
| |
| |
| |
| |
| |
| |
| |
| private $valid_font_display = array( 'auto', 'block', 'fallback', 'swap', 'optional' ); |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| private $style_tag_attrs = array(); |
|
|
| |
| |
| |
| |
| |
| public function __construct() { |
| if ( |
| function_exists( 'is_admin' ) && ! is_admin() |
| && |
| function_exists( 'current_theme_supports' ) && ! current_theme_supports( 'html5', 'style' ) |
| ) { |
| $this->style_tag_attrs = array( 'type' => 'text/css' ); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| public function generate_and_print( array $fonts ) { |
| $fonts = $this->validate_fonts( $fonts ); |
|
|
| |
| if ( empty( $fonts ) ) { |
| return; |
| } |
|
|
| $css = $this->get_css( $fonts ); |
|
|
| |
| |
| |
| |
| |
| $css = wp_strip_all_tags( $css ); |
|
|
| |
| if ( empty( $css ) ) { |
| return; |
| } |
|
|
| printf( $this->get_style_element(), $css ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function validate_fonts( array $fonts ) { |
| $validated_fonts = array(); |
|
|
| foreach ( $fonts as $font_faces ) { |
| foreach ( $font_faces as $font_face ) { |
| $font_face = $this->validate_font_face_declarations( $font_face ); |
| |
| if ( false === $font_face ) { |
| continue; |
| } |
|
|
| $validated_fonts[] = $font_face; |
| } |
| } |
|
|
| return $validated_fonts; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function validate_font_face_declarations( array $font_face ) { |
| $font_face = wp_parse_args( $font_face, $this->font_face_property_defaults ); |
|
|
| |
| if ( empty( $font_face['font-family'] ) || ! is_string( $font_face['font-family'] ) ) { |
| |
| _doing_it_wrong( |
| __METHOD__, |
| __( 'Font font-family must be a non-empty string.' ), |
| '6.4.0' |
| ); |
| return false; |
| } |
|
|
| |
| if ( empty( $font_face['src'] ) || ( ! is_string( $font_face['src'] ) && ! is_array( $font_face['src'] ) ) ) { |
| |
| _doing_it_wrong( |
| __METHOD__, |
| __( 'Font src must be a non-empty string or an array of strings.' ), |
| '6.4.0' |
| ); |
| return false; |
| } |
|
|
| |
| foreach ( (array) $font_face['src'] as $src ) { |
| if ( empty( $src ) || ! is_string( $src ) ) { |
| |
| _doing_it_wrong( |
| __METHOD__, |
| __( 'Each font src must be a non-empty string.' ), |
| '6.4.0' |
| ); |
| return false; |
| } |
| } |
|
|
| |
| if ( ! is_string( $font_face['font-weight'] ) && ! is_int( $font_face['font-weight'] ) ) { |
| |
| _doing_it_wrong( |
| __METHOD__, |
| __( 'Font font-weight must be a properly formatted string or integer.' ), |
| '6.4.0' |
| ); |
| return false; |
| } |
|
|
| |
| if ( ! in_array( $font_face['font-display'], $this->valid_font_display, true ) ) { |
| $font_face['font-display'] = $this->font_face_property_defaults['font-display']; |
| } |
|
|
| |
| foreach ( $font_face as $property => $value ) { |
| if ( ! in_array( $property, $this->valid_font_face_properties, true ) ) { |
| unset( $font_face[ $property ] ); |
| } |
| } |
|
|
| return $font_face; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| private function get_style_element() { |
| $attributes = $this->generate_style_element_attributes(); |
|
|
| return "<style id='wp-fonts-local'{$attributes}>\n%s\n</style>\n"; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| private function generate_style_element_attributes() { |
| $attributes = ''; |
| foreach ( $this->style_tag_attrs as $name => $value ) { |
| $attributes .= " {$name}='{$value}'"; |
| } |
| return $attributes; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| private function get_css( $font_faces ) { |
| $css = ''; |
|
|
| foreach ( $font_faces as $font_face ) { |
| |
| $font_face = $this->order_src( $font_face ); |
|
|
| |
| $css .= '@font-face{' . $this->build_font_face_css( $font_face ) . '}' . "\n"; |
| } |
|
|
| |
| return rtrim( $css, "\n" ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function order_src( array $font_face ) { |
| if ( ! is_array( $font_face['src'] ) ) { |
| $font_face['src'] = (array) $font_face['src']; |
| } |
|
|
| $src = array(); |
| $src_ordered = array(); |
|
|
| foreach ( $font_face['src'] as $url ) { |
| |
| if ( str_starts_with( trim( $url ), 'data:' ) ) { |
| $src_ordered[] = array( |
| 'url' => $url, |
| 'format' => 'data', |
| ); |
| continue; |
| } |
| $format = pathinfo( $url, PATHINFO_EXTENSION ); |
| $src[ $format ] = $url; |
| } |
|
|
| |
| if ( ! empty( $src['woff2'] ) ) { |
| $src_ordered[] = array( |
| 'url' => $src['woff2'], |
| 'format' => 'woff2', |
| ); |
| } |
|
|
| |
| if ( ! empty( $src['woff'] ) ) { |
| $src_ordered[] = array( |
| 'url' => $src['woff'], |
| 'format' => 'woff', |
| ); |
| } |
|
|
| |
| if ( ! empty( $src['ttf'] ) ) { |
| $src_ordered[] = array( |
| 'url' => $src['ttf'], |
| 'format' => 'truetype', |
| ); |
| } |
|
|
| |
| if ( ! empty( $src['eot'] ) ) { |
| $src_ordered[] = array( |
| 'url' => $src['eot'], |
| 'format' => 'embedded-opentype', |
| ); |
| } |
|
|
| |
| if ( ! empty( $src['otf'] ) ) { |
| $src_ordered[] = array( |
| 'url' => $src['otf'], |
| 'format' => 'opentype', |
| ); |
| } |
| $font_face['src'] = $src_ordered; |
|
|
| return $font_face; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function build_font_face_css( array $font_face ) { |
| $css = ''; |
|
|
| |
| |
| |
| |
| if ( |
| str_contains( $font_face['font-family'], ' ' ) && |
| ! str_contains( $font_face['font-family'], '"' ) && |
| ! str_contains( $font_face['font-family'], "'" ) |
| ) { |
| $font_face['font-family'] = '"' . $font_face['font-family'] . '"'; |
| } |
|
|
| foreach ( $font_face as $key => $value ) { |
| |
| if ( 'src' === $key ) { |
| $value = $this->compile_src( $value ); |
| } |
|
|
| |
| if ( 'font-variation-settings' === $key && is_array( $value ) ) { |
| $value = $this->compile_variations( $value ); |
| } |
|
|
| if ( ! empty( $value ) ) { |
| $css .= "$key:$value;"; |
| } |
| } |
|
|
| return $css; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function compile_src( array $value ) { |
| $src = ''; |
|
|
| foreach ( $value as $item ) { |
| $src .= ( 'data' === $item['format'] ) |
| ? ", url({$item['url']})" |
| : ", url('{$item['url']}') format('{$item['format']}')"; |
| } |
|
|
| $src = ltrim( $src, ', ' ); |
| return $src; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| private function compile_variations( array $font_variation_settings ) { |
| $variations = ''; |
|
|
| foreach ( $font_variation_settings as $key => $value ) { |
| $variations .= "$key $value"; |
| } |
|
|
| return $variations; |
| } |
| } |
|
|