summaryrefslogtreecommitdiffstats
path: root/wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php
diff options
context:
space:
mode:
Diffstat (limited to 'wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php')
-rw-r--r--wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php317
1 files changed, 317 insertions, 0 deletions
diff --git a/wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php b/wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php
new file mode 100644
index 0000000..1bf1032
--- /dev/null
+++ b/wp-includes/sitemaps/class-wp-sitemaps-stylesheet.php
@@ -0,0 +1,317 @@
+<?php
+/**
+ * Sitemaps: WP_Sitemaps_Stylesheet class
+ *
+ * This class provides the XSL stylesheets to style all sitemaps.
+ *
+ * @package WordPress
+ * @subpackage Sitemaps
+ * @since 5.5.0
+ */
+
+/**
+ * Stylesheet provider class.
+ *
+ * @since 5.5.0
+ */
+#[AllowDynamicProperties]
+class WP_Sitemaps_Stylesheet {
+ /**
+ * Renders the XSL stylesheet depending on whether it's the sitemap index or not.
+ *
+ * @param string $type Stylesheet type. Either 'sitemap' or 'index'.
+ */
+ public function render_stylesheet( $type ) {
+ header( 'Content-Type: application/xml; charset=UTF-8' );
+
+ if ( 'sitemap' === $type ) {
+ // All content is escaped below.
+ echo $this->get_sitemap_stylesheet();
+ }
+
+ if ( 'index' === $type ) {
+ // All content is escaped below.
+ echo $this->get_sitemap_index_stylesheet();
+ }
+
+ exit;
+ }
+
+ /**
+ * Returns the escaped XSL for all sitemaps, except index.
+ *
+ * @since 5.5.0
+ */
+ public function get_sitemap_stylesheet() {
+ $css = $this->get_stylesheet_css();
+ $title = esc_xml( __( 'XML Sitemap' ) );
+ $description = esc_xml( __( 'This XML Sitemap is generated by WordPress to make your content more visible for search engines.' ) );
+ $learn_more = sprintf(
+ '<a href="%s">%s</a>',
+ esc_url( __( 'https://www.sitemaps.org/' ) ),
+ esc_xml( __( 'Learn more about XML sitemaps.' ) )
+ );
+
+ $text = sprintf(
+ /* translators: %s: Number of URLs. */
+ esc_xml( __( 'Number of URLs in this XML Sitemap: %s.' ) ),
+ '<xsl:value-of select="count( sitemap:urlset/sitemap:url )" />'
+ );
+
+ $lang = get_language_attributes( 'html' );
+ $url = esc_xml( __( 'URL' ) );
+ $lastmod = esc_xml( __( 'Last Modified' ) );
+ $changefreq = esc_xml( __( 'Change Frequency' ) );
+ $priority = esc_xml( __( 'Priority' ) );
+
+ $xsl_content = <<<XSL
+<?xml version="1.0" encoding="UTF-8"?>
+<xsl:stylesheet
+ version="1.0"
+ xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
+ xmlns:sitemap="http://www.sitemaps.org/schemas/sitemap/0.9"
+ exclude-result-prefixes="sitemap"
+ >
+
+ <xsl:output method="html" encoding="UTF-8" indent="yes" />
+
+ <!--
+ Set variables for whether lastmod, changefreq or priority occur for any url in the sitemap.
+ We do this up front because it can be expensive in a large sitemap.
+ -->
+ <xsl:variable name="has-lastmod" select="count( /sitemap:urlset/sitemap:url/sitemap:lastmod )" />
+ <xsl:variable name="has-changefreq" select="count( /sitemap:urlset/sitemap:url/sitemap:changefreq )" />
+ <xsl:variable name="has-priority" select="count( /sitemap:urlset/sitemap:url/sitemap:priority )" />
+
+ <xsl:template match="/">
+ <html {$lang}>
+ <head>
+ <title>{$title}</title>
+ <style>
+ {$css}
+ </style>
+ </head>
+ <body>
+ <div id="sitemap">
+ <div id="sitemap__header">
+ <h1>{$title}</h1>
+ <p>{$description}</p>
+ <p>{$learn_more}</p>
+ </div>
+ <div id="sitemap__content">
+ <p class="text">{$text}</p>
+ <table id="sitemap__table">
+ <thead>
+ <tr>
+ <th class="loc">{$url}</th>
+ <xsl:if test="\$has-lastmod">
+ <th class="lastmod">{$lastmod}</th>
+ </xsl:if>
+ <xsl:if test="\$has-changefreq">
+ <th class="changefreq">{$changefreq}</th>
+ </xsl:if>
+ <xsl:if test="\$has-priority">
+ <th class="priority">{$priority}</th>
+ </xsl:if>
+ </tr>
+ </thead>
+ <tbody>
+ <xsl:for-each select="sitemap:urlset/sitemap:url">
+ <tr>
+ <td class="loc"><a href="{sitemap:loc}"><xsl:value-of select="sitemap:loc" /></a></td>
+ <xsl:if test="\$has-lastmod">
+ <td class="lastmod"><xsl:value-of select="sitemap:lastmod" /></td>
+ </xsl:if>
+ <xsl:if test="\$has-changefreq">
+ <td class="changefreq"><xsl:value-of select="sitemap:changefreq" /></td>
+ </xsl:if>
+ <xsl:if test="\$has-priority">
+ <td class="priority"><xsl:value-of select="sitemap:priority" /></td>
+ </xsl:if>
+ </tr>
+ </xsl:for-each>
+ </tbody>
+ </table>
+ </div>
+ </div>
+ </body>
+ </html>
+ </xsl:template>
+</xsl:stylesheet>
+
+XSL;
+
+ /**
+ * Filters the content of the sitemap stylesheet.
+ *
+ * @since 5.5.0
+ *
+ * @param string $xsl_content Full content for the XML stylesheet.
+ */
+ return apply_filters( 'wp_sitemaps_stylesheet_content', $xsl_content );
+ }
+
+ /**
+ * Returns the escaped XSL for the index sitemaps.
+ *
+ * @since 5.5.0
+ */
+ public function get_sitemap_index_stylesheet() {
+ $css = $this->get_stylesheet_css();
+ $title = esc_xml( __( 'XML Sitemap' ) );
+ $description = esc_xml( __( 'This XML Sitemap is generated by WordPress to make your content more visible for search engines.' ) );
+ $learn_more = sprintf(
+ '<a href="%s">%s</a>',
+ esc_url( __( 'https://www.sitemaps.org/' ) ),
+ esc_xml( __( 'Learn more about XML sitemaps.' ) )
+ );
+
+ $text = sprintf(
+ /* translators: %s: Number of URLs. */
+ esc_xml( __( 'Number of URLs in this XML Sitemap: %s.' ) ),
+ '<xsl:value-of select="count( sitemap:sitemapindex/sitemap:sitemap )" />'
+ );
+
+ $lang = get_language_attributes( 'html' );
+ $url = esc_xml( __( 'URL' ) );
+ $lastmod = esc_xml( __( 'Last Modified' ) );
+
+ $xsl_content = <<<XSL
+<?xml version="1.0" encoding="UTF-8"?>
+<xsl:stylesheet
+ version="1.0"
+ xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
+ xmlns:sitemap="http://www.sitemaps.org/schemas/sitemap/0.9"
+ exclude-result-prefixes="sitemap"
+ >
+
+ <xsl:output method="html" encoding="UTF-8" indent="yes" />
+
+ <!--
+ Set variables for whether lastmod occurs for any sitemap in the index.
+ We do this up front because it can be expensive in a large sitemap.
+ -->
+ <xsl:variable name="has-lastmod" select="count( /sitemap:sitemapindex/sitemap:sitemap/sitemap:lastmod )" />
+
+ <xsl:template match="/">
+ <html {$lang}>
+ <head>
+ <title>{$title}</title>
+ <style>
+ {$css}
+ </style>
+ </head>
+ <body>
+ <div id="sitemap">
+ <div id="sitemap__header">
+ <h1>{$title}</h1>
+ <p>{$description}</p>
+ <p>{$learn_more}</p>
+ </div>
+ <div id="sitemap__content">
+ <p class="text">{$text}</p>
+ <table id="sitemap__table">
+ <thead>
+ <tr>
+ <th class="loc">{$url}</th>
+ <xsl:if test="\$has-lastmod">
+ <th class="lastmod">{$lastmod}</th>
+ </xsl:if>
+ </tr>
+ </thead>
+ <tbody>
+ <xsl:for-each select="sitemap:sitemapindex/sitemap:sitemap">
+ <tr>
+ <td class="loc"><a href="{sitemap:loc}"><xsl:value-of select="sitemap:loc" /></a></td>
+ <xsl:if test="\$has-lastmod">
+ <td class="lastmod"><xsl:value-of select="sitemap:lastmod" /></td>
+ </xsl:if>
+ </tr>
+ </xsl:for-each>
+ </tbody>
+ </table>
+ </div>
+ </div>
+ </body>
+ </html>
+ </xsl:template>
+</xsl:stylesheet>
+
+XSL;
+
+ /**
+ * Filters the content of the sitemap index stylesheet.
+ *
+ * @since 5.5.0
+ *
+ * @param string $xsl_content Full content for the XML stylesheet.
+ */
+ return apply_filters( 'wp_sitemaps_stylesheet_index_content', $xsl_content );
+ }
+
+ /**
+ * Gets the CSS to be included in sitemap XSL stylesheets.
+ *
+ * @since 5.5.0
+ *
+ * @return string The CSS.
+ */
+ public function get_stylesheet_css() {
+ $text_align = is_rtl() ? 'right' : 'left';
+
+ $css = <<<EOF
+
+ body {
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif;
+ color: #444;
+ }
+
+ #sitemap {
+ max-width: 980px;
+ margin: 0 auto;
+ }
+
+ #sitemap__table {
+ width: 100%;
+ border: solid 1px #ccc;
+ border-collapse: collapse;
+ }
+
+ #sitemap__table tr td.loc {
+ /*
+ * URLs should always be LTR.
+ * See https://core.trac.wordpress.org/ticket/16834
+ * and https://core.trac.wordpress.org/ticket/49949
+ */
+ direction: ltr;
+ }
+
+ #sitemap__table tr th {
+ text-align: {$text_align};
+ }
+
+ #sitemap__table tr td,
+ #sitemap__table tr th {
+ padding: 10px;
+ }
+
+ #sitemap__table tr:nth-child(odd) td {
+ background-color: #eee;
+ }
+
+ a:hover {
+ text-decoration: none;
+ }
+
+EOF;
+
+ /**
+ * Filters the CSS only for the sitemap stylesheet.
+ *
+ * @since 5.5.0
+ *
+ * @param string $css CSS to be applied to default XSL file.
+ */
+ return apply_filters( 'wp_sitemaps_stylesheet_css', $css );
+ }
+}