plugins); wp_send_json_success($html); return; } catch (Exception $e) { error_log('Error displaying cached plugins: ' . $e->getMessage()); // Fall through to fetch fresh data } } error_log('Fetching fresh data for category: ' . $category); error_log('Plugins to fetch: ' . implode(', ', $recommended_plugins[$category])); try { $plugins = array(); // Only fetch plugins that are in our recommended list for this category foreach ($recommended_plugins[$category] as $slug) { try { error_log('Fetching plugin data for: ' . $slug); $plugin_data = plugins_api('plugin_information', array( 'slug' => $slug, 'fields' => array( 'short_description' => true, 'sections' => false, 'requires' => true, 'rating' => true, 'ratings' => false, 'downloaded' => true, 'last_updated' => true, 'added' => false, 'tags' => false, 'compatibility' => false, 'homepage' => true, 'versions' => false, 'donate_link' => false, 'reviews' => false, 'banners' => false, 'icons' => true, 'active_installs' => true, 'group' => false, 'contributors' => false, ) )); if (is_wp_error($plugin_data)) { error_log('Error fetching plugin data for ' . $slug . ': ' . $plugin_data->get_error_message()); } else { $plugins[] = $plugin_data; error_log('Successfully fetched data for: ' . $slug); } } catch (Exception $e) { error_log('Exception fetching plugin data for ' . $slug . ': ' . $e->getMessage()); continue; } } if (empty($plugins)) { wp_send_json_error('No plugin data could be retrieved for category: ' . $category); return; } error_log('Total plugins fetched: ' . count($plugins)); // Create response object $res = (object) array( 'info' => array( 'page' => 1, 'pages' => 1, 'results' => count($plugins), ), 'plugins' => $plugins ); // Cache the results wp_allstars_set_cached_plugins($category, $res); // Generate plugin cards HTML $html = wp_allstars_generate_plugin_cards($plugins); wp_send_json_success($html); } catch (Exception $e) { error_log('Failed to fetch plugin data: ' . $e->getMessage()); wp_send_json_error('Failed to fetch plugin data: ' . $e->getMessage()); } } add_action('wp_ajax_wp_allstars_get_plugins', 'wp_allstars_ajax_get_plugins'); // Function to generate plugin cards HTML function wp_allstars_generate_plugin_cards($plugins) { if (empty($plugins)) { return '

No plugins found.

'; } ob_start(); ?>

name); ?>

icons) && !empty($plugin->icons['1x'])): ?>

short_description); ?>

author); ?>

$plugin->rating, 'type' => 'percent', 'number' => $plugin->num_ratings)); ?> (num_ratings); ?>)
last_updated))); ?>
downloaded), number_format_i18n($plugin->downloaded)); ?>
tested) && version_compare(substr($GLOBALS['wp_version'], 0, strlen($plugin->tested)), $plugin->tested, '>')) { echo '' . __('Untested with your version of WordPress') . ''; } elseif (!empty($plugin->requires) && version_compare($GLOBALS['wp_version'], $plugin->requires, '<')) { echo '' . __('Incompatible with your version of WordPress') . ''; } else { echo '' . __('Compatible with your version of WordPress') . ''; } ?>
%s', esc_url(wp_allstars_get_pro_plugin_url($pro_plugin)), esc_html__('Go Pro', 'wp-allstars') ); break; } } return $action_links; } /** * Helper function to get the pro plugin URL from the button_group array * * @param array $pro_plugin The pro plugin configuration array * @return string The URL for the pro plugin */ function wp_allstars_get_pro_plugin_url($pro_plugin) { // Find the primary button URL in the button_group array $pro_url = ''; if (!empty($pro_plugin['button_group'])) { // First try to find a primary button foreach ($pro_plugin['button_group'] as $button) { if (isset($button['primary']) && $button['primary']) { $pro_url = $button['url']; break; } } // If no primary button found, use the first button URL if (empty($pro_url) && !empty($pro_plugin['button_group'][0]['url'])) { $pro_url = $pro_plugin['button_group'][0]['url']; } } return $pro_url; } // Remove the old plugins API filter since we're handling everything in the AJAX endpoint remove_filter('plugins_api_result', 'wp_allstars_plugins_api_result'); // Clear plugin cache when plugins are updated, activated, or deactivated function wp_allstars_clear_plugin_cache() { $recommended_plugins = wp_allstars_get_recommended_plugins(); foreach (array_keys($recommended_plugins) as $category) { delete_transient('wp_allstars_plugins_' . $category); } } add_action('upgrader_process_complete', 'wp_allstars_clear_plugin_cache', 10, 0); add_action('activated_plugin', 'wp_allstars_clear_plugin_cache'); add_action('deactivated_plugin', 'wp_allstars_clear_plugin_cache'); add_action('deleted_plugin', 'wp_allstars_clear_plugin_cache'); add_action('update_option_active_plugins', 'wp_allstars_clear_plugin_cache'); // Add transient caching for theme data function wp_allstars_get_cached_theme() { $cache_key = 'wp_allstars_theme_kadence'; return get_transient($cache_key); } function wp_allstars_set_cached_theme($data) { $cache_key = 'wp_allstars_theme_kadence'; set_transient($cache_key, $data, 12 * HOUR_IN_SECONDS); } // Add AJAX endpoint for theme function wp_allstars_ajax_get_themes() { error_log('WP ALLSTARS: Theme AJAX handler started'); // Check nonce with the correct action name if (!isset($_POST['_wpnonce'])) { error_log('WP ALLSTARS: No nonce provided'); wp_send_json_error('No security token provided.'); return; } if (!check_ajax_referer('wp-allstars-nonce', '_wpnonce', false)) { error_log('WP ALLSTARS: Invalid nonce: ' . sanitize_text_field($_POST['_wpnonce'])); wp_send_json_error('Invalid security token sent.'); return; } if (!current_user_can('install_themes')) { error_log('WP ALLSTARS: User does not have permission to install themes'); wp_send_json_error('Permission denied'); return; } error_log('WP ALLSTARS: Starting theme fetch process'); try { error_log('WP ALLSTARS: Fetching theme data for kadence'); // Check if we have cached data first $theme_data = wp_allstars_get_cached_theme(); // If no cached data, fetch from API if (empty($theme_data)) { error_log('WP ALLSTARS: No cached theme data, fetching from API'); // Get theme data with minimal fields $theme_data = themes_api('theme_information', array( 'slug' => 'kadence', 'fields' => array( 'sections' => false, 'description' => true, 'rating' => true, 'ratings' => false, 'downloaded' => true, 'download_link' => true, 'last_updated' => true, 'homepage' => true, 'tags' => false, 'screenshot_url' => true, 'version' => true, 'requires' => true, 'requires_php' => true, 'active_installs' => true, 'author' => true, 'preview_url' => true, ) )); // Cache the result if successful if (!is_wp_error($theme_data)) { wp_allstars_set_cached_theme($theme_data); } } else { error_log('WP ALLSTARS: Using cached theme data'); } if (is_wp_error($theme_data)) { error_log('WP ALLSTARS Theme API Error: ' . $theme_data->get_error_message()); wp_send_json_error('Theme API Error: ' . $theme_data->get_error_message()); return; } error_log('WP ALLSTARS: Successfully fetched theme data'); // Format author data $author = ''; if (is_string($theme_data->author)) { $author = $theme_data->author; } elseif (is_array($theme_data->author)) { $author = isset($theme_data->author['display_name']) ? $theme_data->author['display_name'] : ''; } error_log('WP ALLSTARS: Theme data retrieved, generating HTML'); // Generate custom HTML for the theme using our template partial ob_start(); include(plugin_dir_path(__FILE__) . 'partials/theme-panel.php'); $html = ob_get_clean(); if (empty($html)) { error_log('WP ALLSTARS: Empty HTML generated'); wp_send_json_error('Failed to generate theme display'); return; } error_log('WP ALLSTARS: Successfully generated theme display, HTML length: ' . strlen($html)); wp_send_json_success($html); exit; // Ensure we exit after sending the JSON response } catch (Exception $e) { error_log('WP ALLSTARS Theme loading exception: ' . $e->getMessage()); error_log('WP ALLSTARS Theme loading exception trace: ' . $e->getTraceAsString()); wp_send_json_error('Theme loading error: ' . $e->getMessage()); } catch (Error $e) { error_log('WP ALLSTARS Theme loading error: ' . $e->getMessage()); error_log('WP ALLSTARS Theme loading error trace: ' . $e->getTraceAsString()); wp_send_json_error('Theme loading error: ' . $e->getMessage()); } } add_action('wp_ajax_wp_allstars_get_themes', 'wp_allstars_ajax_get_themes'); // Clear theme cache when themes are updated function wp_allstars_clear_theme_cache() { delete_transient('wp_allstars_theme_kadence'); } add_action('upgrader_process_complete', 'wp_allstars_clear_theme_cache', 10, 0); add_action('switch_theme', 'wp_allstars_clear_theme_cache'); // Add AJAX handler for theme activation function wp_allstars_activate_theme() { // Debug information error_log('Theme activation AJAX request received: ' . print_r($_POST, true)); // Check nonce with the correct action name if (!check_ajax_referer('wp-allstars-nonce', '_wpnonce', false)) { error_log('Theme activation failed: Invalid nonce'); wp_send_json_error('Invalid security token sent.'); return; } if (!current_user_can('switch_themes')) { error_log('Theme activation failed: Permission denied'); wp_send_json_error('Permission denied'); return; } $theme = isset($_POST['theme']) ? sanitize_text_field($_POST['theme']) : ''; if (empty($theme)) { error_log('Theme activation failed: No theme specified'); wp_send_json_error('No theme specified'); return; } // Get the theme object $theme_obj = wp_get_theme($theme); if (!$theme_obj->exists()) { error_log('Theme activation failed: Theme does not exist - ' . $theme); wp_send_json_error('Theme does not exist'); return; } if ($theme_obj->errors()) { error_log('Theme activation failed: Theme has errors - ' . print_r($theme_obj->errors(), true)); wp_send_json_error('Theme has errors'); return; } // Check if theme is already active $current_theme = wp_get_theme(); if ($current_theme->get_stylesheet() === $theme) { error_log('Theme is already active: ' . $theme); wp_send_json_success(array( 'message' => 'Theme is already active', 'customize_url' => admin_url('customize.php') )); return; } // Switch the theme error_log('Switching to theme: ' . $theme); switch_theme($theme); // Check if the switch was successful $active_theme = wp_get_theme(); if ($active_theme->get_stylesheet() === $theme) { error_log('Theme activated successfully: ' . $theme); wp_send_json_success(array( 'message' => 'Theme activated successfully', 'customize_url' => admin_url('customize.php') )); } else { error_log('Failed to activate theme: ' . $theme . ', active theme is: ' . $active_theme->get_stylesheet()); wp_send_json_error('Failed to activate theme'); } } add_action('wp_ajax_wp_allstars_activate_theme', 'wp_allstars_activate_theme'); // Register settings page HTML function wp_allstars_settings_page() { global $tabs; $active_tab = isset($_GET['tab']) ? $_GET['tab'] : 'general'; $active_category = isset($_GET['category']) ? $_GET['category'] : 'minimal'; // Clear cache and load required files if ($active_tab === 'recommended') { wp_allstars_clear_plugin_cache(); require_once ABSPATH . 'wp-admin/includes/plugin-install.php'; wp_enqueue_script('plugin-install'); wp_enqueue_script('updates'); add_thickbox(); wp_enqueue_style('wp-allstars-admin', plugins_url('css/wp-allstars-admin.css', __FILE__)); wp_enqueue_style('wp-allstars-plugins', plugins_url('css/wp-allstars-plugins.css', __FILE__)); // Add inline script to load plugins on page load wp_add_inline_script('wp-allstars-admin', ' jQuery(document).ready(function($) { if ($("#wpa-plugin-list").length && $("#wpa-plugin-list").is(":empty")) { var category = "' . esc_js($active_category) . '"; var $container = $("#wpa-plugin-list"); var $loadingOverlay = $("
"); // Show loading overlay $container.css("position", "relative").append($loadingOverlay); // AJAX request to get plugins $.ajax({ url: ajaxurl, type: "POST", data: { action: "wp_allstars_get_plugins", category: category, _wpnonce: wpAllstars.nonce }, success: function(response) { $loadingOverlay.remove(); if (response.success) { $container.html(response.data); // Initialize plugin action buttons if (typeof initPluginActions === "function") { initPluginActions(); } // Spinners have been removed from individual cards } else { $container.html("

" + response.data + "

"); } }, error: function(xhr, status, error) { $loadingOverlay.remove(); $container.html("

Failed to load plugins. Please try again. Error: " + error + "

"); console.error("AJAX Error:", xhr.responseText); } }); } }); '); } elseif ($active_tab === 'theme') { wp_allstars_clear_theme_cache(); require_once ABSPATH . 'wp-admin/includes/theme.php'; wp_enqueue_script('theme-install'); wp_enqueue_script('updates'); add_thickbox(); wp_enqueue_style('wp-allstars-admin', plugins_url('css/wp-allstars-admin.css', __FILE__)); wp_enqueue_style('wp-allstars-plugins', plugins_url('css/wp-allstars-plugins.css', __FILE__)); // Add inline script to load themes directly - same approach as plugins tab wp_add_inline_script('wp-allstars-admin', ' jQuery(document).ready(function($) { if ($("#wpa-theme-list").length && $("#wpa-theme-list").is(":empty")) { var $container = $("#wpa-theme-list"); var $loadingOverlay = $("
"); // Show loading overlay $container.css("position", "relative").append($loadingOverlay); // AJAX request to get themes $.ajax({ url: ajaxurl, type: "POST", data: { action: "wp_allstars_get_themes", _wpnonce: wpAllstars.nonce }, success: function(response) { $loadingOverlay.remove(); if (response.success) { $container.html(response.data); // Initialize theme action buttons if (typeof initThemeHandlers === "function") { initThemeHandlers(); } } else { $container.html("

" + response.data + "

"); } }, error: function(xhr, status, error) { $loadingOverlay.remove(); $container.html("

Failed to load themes. Please try again. Error: " + error + "

"); console.error("AJAX Error:", xhr.responseText); } }); } }); '); } ?>

Version

%filename%, %post_id%, %postname%, %timestamp%, %date%, %year%, %month%, %day%

%filename%, %post_title%, %post_id%, %postname%, %timestamp%

/>

wp_create_nonce('wp-allstars-nonce'), 'ajaxurl' => admin_url('admin-ajax.php') )); } add_action('admin_enqueue_scripts', 'wp_allstars_admin_enqueue_scripts');