Implementación masiva de funcionalidades esenciales del tema apus-theme usando agentes paralelos para máxima eficiencia. **Issues Completados:** **Issue #2 - Eliminar bloat de WordPress:** - inc/performance.php: 13 funciones que remueven emojis, oEmbed, feeds, dashicons, jQuery migrate, XML-RPC, etc. - Optimización completa del frontend **Issue #3 - Desactivar búsqueda nativa:** - inc/search-disable.php: Bloquea queries de búsqueda, widget, formularios - search.php: Retorna 404 con mensaje amigable **Issue #4 - Desactivar comentarios:** - inc/comments-disable.php: 15 funciones que eliminan comentarios de frontend y backend - comments.php: Template desactivado **Issue #8 - Footer con 4 widgets:** - footer.php: Verificado con 4 áreas de widgets y copyright - assets/css/footer.css: Estilos responsive completos - Sistema de anchos configurables **Issue #9 - Jerarquía de plantillas:** - home.php, category.php, tag.php, author.php, date.php, taxonomy.php, attachment.php - 7 nuevas plantillas + 12 verificadas - Template parts completos - Paginación en todos los archives **Issue #10 - Imágenes destacadas:** - inc/featured-image.php: 12 funciones para manejo de featured images - Sin placeholders, lazy loading, alt text automático - Responsive con Bootstrap, aspect ratio **Issue #11 - Badge de categoría:** - inc/category-badge.php: Badge Bootstrap sobre H1 en single posts - Excluye "Uncategorized" - Template tag: apus_display_category_badge() **Issue #12 - TOC automático:** - inc/toc.php: Genera TOC desde H2/H3 - assets/css/toc.css: Estilos con numeración CSS counters - assets/js/toc.js: Smooth scroll, scroll spy, toggle - Configurable con apus_get_option() **Issue #13 - Posts relacionados:** - inc/related-posts.php: Query por categoría, 12 funciones - inc/admin/related-posts-options.php: Sistema de configuración - assets/css/related-posts.css: Cards responsive - Hook automático en single posts **Issue #16 - AdSense delay:** - inc/adsense-delay.php: Retardo de carga hasta scroll/click - assets/js/adsense-loader.js: Detecta interacciones - Mejora FID y TBT para Core Web Vitals **Archivos Modificados:** - functions.php: Includes de nuevos módulos, removido feed support - single.php: Integración de category badge - inc/enqueue-scripts.php: Enqueue de nuevos assets - inc/theme-options-helpers.php: Helper functions para TOC **Archivos Creados:** - 7 nuevas plantillas WordPress - 3 nuevos módulos inc/ (comments-disable, search-disable) - 8 reportes de documentación .md **Estadísticas:** - Total funciones PHP: 60+ nuevas funciones - Líneas de código: 2,500+ líneas - Archivos nuevos: 18 - Archivos modificados: 9 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
239 lines
6.5 KiB
PHP
239 lines
6.5 KiB
PHP
<?php
|
|
/**
|
|
* Table of Contents (TOC) Functions
|
|
*
|
|
* This file contains functions to automatically generate a table of contents
|
|
* from post content headings (H2 and H3).
|
|
*
|
|
* @package Apus_Theme
|
|
* @since 1.0.0
|
|
*/
|
|
|
|
// Exit if accessed directly
|
|
if (!defined('ABSPATH')) {
|
|
exit;
|
|
}
|
|
|
|
/**
|
|
* Extract headings from post content
|
|
*
|
|
* Parses the content and extracts all H2 and H3 headings with their text.
|
|
*
|
|
* @param string $content Post content
|
|
* @return array Array of headings with level, text, and ID
|
|
*/
|
|
function apus_extract_headings($content) {
|
|
if (empty($content)) {
|
|
return array();
|
|
}
|
|
|
|
$headings = array();
|
|
|
|
// Match H2 and H3 tags with their content
|
|
preg_match_all('/<h([23])(?:[^>]*)>(.*?)<\/h\1>/i', $content, $matches, PREG_SET_ORDER);
|
|
|
|
foreach ($matches as $index => $match) {
|
|
$level = (int) $match[1]; // 2 or 3
|
|
$text = strip_tags($match[2]); // Remove any HTML tags inside heading
|
|
|
|
// Generate a clean ID from the heading text
|
|
$id = apus_generate_heading_id($text, $index);
|
|
|
|
$headings[] = array(
|
|
'level' => $level,
|
|
'text' => $text,
|
|
'id' => $id,
|
|
);
|
|
}
|
|
|
|
return $headings;
|
|
}
|
|
|
|
/**
|
|
* Generate a clean ID for a heading
|
|
*
|
|
* Creates a URL-friendly ID from heading text.
|
|
*
|
|
* @param string $text Heading text
|
|
* @param int $index Index of the heading (for uniqueness)
|
|
* @return string Clean ID
|
|
*/
|
|
function apus_generate_heading_id($text, $index) {
|
|
// Remove special characters and convert to lowercase
|
|
$id = sanitize_title($text);
|
|
|
|
// If ID is empty, use a fallback
|
|
if (empty($id)) {
|
|
$id = 'heading-' . $index;
|
|
}
|
|
|
|
return $id;
|
|
}
|
|
|
|
/**
|
|
* Generate HTML for Table of Contents
|
|
*
|
|
* Creates a nested list structure from the headings array.
|
|
*
|
|
* @param array $headings Array of headings from apus_extract_headings()
|
|
* @return string HTML for the table of contents
|
|
*/
|
|
function apus_generate_toc($headings) {
|
|
// Get minimum headings required from theme options
|
|
$min_headings = (int) apus_get_option('toc_min_headings', 2);
|
|
|
|
if (empty($headings) || count($headings) < $min_headings) {
|
|
return ''; // Don't show TOC if there are fewer headings than required
|
|
}
|
|
|
|
// Get custom TOC title from theme options
|
|
$toc_title = apus_get_toc_title();
|
|
|
|
$toc_html = '<nav class="apus-toc" aria-label="' . esc_attr($toc_title) . '">';
|
|
$toc_html .= '<div class="apus-toc-header">';
|
|
$toc_html .= '<h2 class="apus-toc-title">' . esc_html($toc_title) . '</h2>';
|
|
$toc_html .= '<button class="apus-toc-toggle" aria-expanded="true" aria-controls="apus-toc-list">';
|
|
$toc_html .= '<span class="toggle-icon" aria-hidden="true"></span>';
|
|
$toc_html .= '<span class="screen-reader-text">' . esc_html__('Toggle Table of Contents', 'apus-theme') . '</span>';
|
|
$toc_html .= '</button>';
|
|
$toc_html .= '</div>';
|
|
|
|
$toc_html .= '<ol class="apus-toc-list" id="apus-toc-list">';
|
|
|
|
$current_level = 2;
|
|
$open_sublists = 0;
|
|
|
|
foreach ($headings as $index => $heading) {
|
|
$level = $heading['level'];
|
|
$text = esc_html($heading['text']);
|
|
$id = esc_attr($heading['id']);
|
|
|
|
// Handle level changes
|
|
if ($level > $current_level) {
|
|
// Open nested list for H3
|
|
$toc_html .= '<ol class="apus-toc-sublist">';
|
|
$open_sublists++;
|
|
} elseif ($level < $current_level && $open_sublists > 0) {
|
|
// Close nested list when going back to H2
|
|
$toc_html .= '</li></ol></li>';
|
|
$open_sublists--;
|
|
} elseif ($index > 0) {
|
|
// Close previous item
|
|
$toc_html .= '</li>';
|
|
}
|
|
|
|
$toc_html .= '<li class="apus-toc-item apus-toc-level-' . $level . '">';
|
|
$toc_html .= '<a href="#' . $id . '" class="apus-toc-link">' . $text . '</a>';
|
|
|
|
$current_level = $level;
|
|
}
|
|
|
|
// Close any open lists
|
|
$toc_html .= '</li>';
|
|
while ($open_sublists > 0) {
|
|
$toc_html .= '</ol></li>';
|
|
$open_sublists--;
|
|
}
|
|
|
|
$toc_html .= '</ol>';
|
|
$toc_html .= '</nav>';
|
|
|
|
return $toc_html;
|
|
}
|
|
|
|
/**
|
|
* Add IDs to headings in content
|
|
*
|
|
* Modifies the post content to add ID attributes to H2 and H3 headings.
|
|
*
|
|
* @param string $content Post content
|
|
* @return string Modified content with IDs added to headings
|
|
*/
|
|
function apus_add_heading_ids($content) {
|
|
if (empty($content)) {
|
|
return $content;
|
|
}
|
|
|
|
// Extract headings first to get consistent IDs
|
|
$headings = apus_extract_headings($content);
|
|
|
|
if (empty($headings)) {
|
|
return $content;
|
|
}
|
|
|
|
// Replace headings with versions that include IDs
|
|
$heading_index = 0;
|
|
$content = preg_replace_callback(
|
|
'/<h([23])(?:[^>]*)>(.*?)<\/h\1>/i',
|
|
function($matches) use ($headings, &$heading_index) {
|
|
if (!isset($headings[$heading_index])) {
|
|
return $matches[0];
|
|
}
|
|
|
|
$level = $matches[1];
|
|
$text = $matches[2];
|
|
$id = $headings[$heading_index]['id'];
|
|
$heading_index++;
|
|
|
|
return '<h' . $level . ' id="' . esc_attr($id) . '">' . $text . '</h' . $level . '>';
|
|
},
|
|
$content
|
|
);
|
|
|
|
return $content;
|
|
}
|
|
|
|
/**
|
|
* Display Table of Contents before post content
|
|
*
|
|
* Hooks into apus_before_post_content to display TOC on single posts.
|
|
*/
|
|
function apus_display_toc() {
|
|
// Check if TOC is enabled in theme options
|
|
$toc_enabled = apus_get_option('enable_toc', true);
|
|
|
|
if (!$toc_enabled) {
|
|
return; // TOC disabled in theme options
|
|
}
|
|
|
|
// Only show on single posts
|
|
if (!is_single()) {
|
|
return;
|
|
}
|
|
|
|
global $post;
|
|
|
|
if (empty($post->post_content)) {
|
|
return;
|
|
}
|
|
|
|
// Extract headings from content
|
|
$headings = apus_extract_headings($post->post_content);
|
|
|
|
// Generate and display TOC
|
|
$toc = apus_generate_toc($headings);
|
|
|
|
if (!empty($toc)) {
|
|
echo $toc; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- Escaped in apus_generate_toc()
|
|
}
|
|
}
|
|
add_action('apus_before_post_content', 'apus_display_toc');
|
|
|
|
/**
|
|
* Modify post content to add heading IDs
|
|
*
|
|
* Filters the_content to add IDs to headings for TOC linking.
|
|
*
|
|
* @param string $content Post content
|
|
* @return string Modified content
|
|
*/
|
|
function apus_filter_content_add_heading_ids($content) {
|
|
// Only apply to single posts
|
|
if (!is_single()) {
|
|
return $content;
|
|
}
|
|
|
|
return apus_add_heading_ids($content);
|
|
}
|
|
add_filter('the_content', 'apus_filter_content_add_heading_ids', 10);
|