Backup pre-corrección namespaces: mejoras schemas y componentes

Cambios incluidos:
- Actualización de copy/textos en 7 schemas JSON
- Mejoras en AdminAjaxHandler con mapeos adicionales
- Refactorización de FormBuilders y Renderers
- Correcciones en dashboard admin JS
- Nuevo ContactFormRenderer funcional

NOTA: Este commit sirve como respaldo antes de corregir
inconsistencias de case en namespaces (API→Api, WordPress→Wordpress)

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
FrankZamora
2025-11-26 17:59:01 -06:00
parent 0846a3bf03
commit 4c807e1cf2
26 changed files with 717 additions and 79 deletions

View File

@@ -58,8 +58,11 @@ final class NewsletterAjaxHandler
return;
}
// 3. Validar email
// 3. Validar y sanitizar campos
$email = sanitize_email($_POST['email'] ?? '');
$name = sanitize_text_field($_POST['name'] ?? '');
$whatsapp = sanitize_text_field($_POST['whatsapp'] ?? '');
if (empty($email) || !is_email($email)) {
wp_send_json_error([
'message' => __('Por favor ingresa un email valido.', 'roi-theme')
@@ -94,12 +97,20 @@ final class NewsletterAjaxHandler
// 5. Preparar payload
$payload = [
'email' => $email,
'name' => $name,
'whatsapp' => $whatsapp,
'source' => 'newsletter-footer',
'pageUrl' => sanitize_url($_POST['pageUrl'] ?? ''),
'pageTitle' => sanitize_text_field($_POST['pageTitle'] ?? ''),
'timestamp' => current_time('c'),
'timezone' => wp_timezone_string(),
'siteName' => get_bloginfo('name'),
'siteUrl' => home_url(),
];
// Debug: Log payload enviado
error_log('ROI Theme Newsletter: Enviando a webhook - ' . wp_json_encode($payload));
// 6. Enviar a webhook
$result = $this->sendToWebhook($webhookUrl, $payload);
@@ -120,6 +131,8 @@ final class NewsletterAjaxHandler
*/
private function sendToWebhook(string $url, array $payload): array
{
error_log('ROI Theme Newsletter: Webhook URL - ' . $url);
$response = wp_remote_post($url, [
'timeout' => 30,
'headers' => [
@@ -130,6 +143,7 @@ final class NewsletterAjaxHandler
]);
if (is_wp_error($response)) {
error_log('ROI Theme Newsletter: WP Error - ' . $response->get_error_message());
return [
'success' => false,
'error' => $response->get_error_message()
@@ -137,6 +151,10 @@ final class NewsletterAjaxHandler
}
$statusCode = wp_remote_retrieve_response_code($response);
$responseBody = wp_remote_retrieve_body($response);
error_log('ROI Theme Newsletter: Response Code - ' . $statusCode);
error_log('ROI Theme Newsletter: Response Body - ' . $responseBody);
if ($statusCode >= 200 && $statusCode < 300) {
return ['success' => true, 'error' => ''];

View File

@@ -156,6 +156,11 @@ final class FooterRenderer implements RendererInterface
'color' => $linkHoverColor,
]);
// Widget 1B spacing
$cssRules[] = $this->cssGenerator->generate('.roi-footer .footer-widget-1b', [
'margin-top' => '1.5rem',
]);
// Newsletter description
$cssRules[] = $this->cssGenerator->generate('.roi-footer .newsletter-description', [
'color' => $textColor,
@@ -248,6 +253,7 @@ final class FooterRenderer implements RendererInterface
private function generateHTML(array $data): string
{
$widget1 = $data['widget_1'] ?? [];
$widget1b = $data['widget_1b'] ?? [];
$widget2 = $data['widget_2'] ?? [];
$widget3 = $data['widget_3'] ?? [];
$newsletter = $data['newsletter'] ?? [];
@@ -259,12 +265,15 @@ final class FooterRenderer implements RendererInterface
$newsletterVisible = $this->toBool($newsletter['newsletter_visible'] ?? true);
$widget1Title = esc_html($widget1['widget_1_title'] ?? 'Recursos');
$widget1bTitle = esc_html($widget1b['widget_1b_title'] ?? 'Bases de datos');
$widget2Title = esc_html($widget2['widget_2_title'] ?? 'Soporte');
$widget3Title = esc_html($widget3['widget_3_title'] ?? 'Empresa');
$newsletterTitle = esc_html($newsletter['newsletter_title'] ?? 'Suscribete al Newsletter');
$newsletterDesc = esc_html($newsletter['newsletter_description'] ?? 'Recibe las ultimas actualizaciones.');
$newsletterPlaceholder = esc_attr($newsletter['newsletter_placeholder'] ?? 'Email');
$newsletterNamePlaceholder = esc_attr($newsletter['newsletter_name_placeholder'] ?? 'Nombre');
$newsletterEmailPlaceholder = esc_attr($newsletter['newsletter_email_placeholder'] ?? 'Email');
$newsletterWhatsappPlaceholder = esc_attr($newsletter['newsletter_whatsapp_placeholder'] ?? 'WhatsApp');
$newsletterBtnText = esc_html($newsletter['newsletter_button_text'] ?? 'Suscribirse');
$copyrightText = esc_html($footerBottom['copyright_text'] ?? date('Y') . ' Todos los derechos reservados.');
@@ -276,12 +285,25 @@ final class FooterRenderer implements RendererInterface
$html .= '<div class="container">';
$html .= '<div class="footer-grid">';
// Widget 1
// Columna 1: Widget 1 + Widget 1B
if ($widget1Visible) {
$html .= '<div class="footer-column footer-column-1">';
// Widget 1
$html .= '<div class="footer-widget footer-widget-menu">';
$html .= '<h5 class="widget-title">' . $widget1Title . '</h5>';
$html .= $this->renderMenu('footer_menu_1');
$html .= '</div>';
// Widget 1B - Solo si tiene menu asignado
if (has_nav_menu('footer_menu_4')) {
$html .= '<div class="footer-widget footer-widget-menu footer-widget-1b">';
$html .= '<h5 class="widget-title">' . $widget1bTitle . '</h5>';
$html .= $this->renderMenu('footer_menu_4');
$html .= '</div>';
}
$html .= '</div>';
}
// Widget 2
@@ -308,7 +330,9 @@ final class FooterRenderer implements RendererInterface
$html .= '<form id="roi-newsletter-form" class="newsletter-form">';
$html .= '<input type="hidden" name="action" value="roi_newsletter_subscribe">';
$html .= '<input type="hidden" name="nonce" value="' . esc_attr($nonce) . '">';
$html .= '<input type="email" name="email" class="newsletter-input" placeholder="' . $newsletterPlaceholder . '" required>';
$html .= '<input type="text" name="name" class="newsletter-input" placeholder="' . $newsletterNamePlaceholder . '">';
$html .= '<input type="email" name="email" class="newsletter-input" placeholder="' . $newsletterEmailPlaceholder . '" required>';
$html .= '<input type="tel" name="whatsapp" class="newsletter-input" placeholder="' . $newsletterWhatsappPlaceholder . '">';
$html .= '<button type="submit" class="newsletter-btn">' . $newsletterBtnText . '</button>';
$html .= '<div class="newsletter-message"></div>';
$html .= '</form>';
@@ -384,6 +408,8 @@ final class FooterRenderer implements RendererInterface
try {
const formData = new FormData(form);
formData.append('pageUrl', window.location.href);
formData.append('pageTitle', document.title);
const response = await fetch('{$ajaxUrl}', {
method: 'POST',