File: /var/www/html/wp-content/plugins/media-cleaner/common/admin.php
<?php
if ( !class_exists( 'MeowKit_WPMC_Admin' ) ) {
class MeowKit_WPMC_Admin {
public static $loaded = false;
public static $version = '5.0';
public static $admin_version = '5.0';
public static $network_license_modal_added = false;
public static $network_license_plugins = [];
/**
* Storage for instances that need deferred initialization.
*
* WordPress Loading Sequence Problem:
* 1. Load all plugin files
* 2. Fire 'plugins_loaded' hook ← Most plugins instantiate Admin here
* 3. Load wp-includes/pluggable.php ← current_user_can() defined here
* 4. Fire 'init' hook ← Safe to use pluggable functions
*
* When plugins instantiate during 'plugins_loaded', the pluggable functions
* (current_user_can, wp_get_current_user) don't exist yet. This array stores
* instances until 'init' when we can safely call those functions.
*
* @var array
*/
private static $deferred_instances = array();
public $prefix; // prefix used for actions, filters (mfrh)
public $mainfile; // plugin main file (media-file-renamer.php)
public $domain; // domain used for translation (media-file-renamer)
public $isPro = false;
// Store constructor params that affect per-instance setup
private $disableReview = false;
public static $logo = 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNjQiIGhlaWdodD0iNDYiIHZpZXdCb3g9IjAgMCA2NCA0NiIgZmlsbD0ibm9uZSIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICA8ZyBjbGlwUGF0aD0idXJsKCNjbGlwMF8zMTBfMjI5KSI+CiAgICA8cGF0aCBkPSJNNjQgMzAuNjQwOEM2NCAyNy43OTg1IDYwLjA4MTYgMjUuODMwMyA1NS44Mjk4IDI1LjgzMDNDNTQuODU5MyAyNS44MzAzIDUzLjkzMTEgMjUuOTMzIDUzLjA3NiAyNi4xMjUzQzQ5Ljg4NjUgMTkuMDc5IDQxLjY1MzkgMTMuMDg1MyAzMi4wMDAyIDEzLjA4NTNDMzAuODMzNyAxMy4wODUzIDI5LjY4ODEgMTMuMTcyNyAyOC41Njk4IDEzLjMzOTJDMjcuMjA2OSAxMC4zMDc2IDIyLjY3NjIgMi40MzQyNiAxMS41OTU0IDAuMDgzMDA2NEMxMS4wNDkxIC0wLjAzMjc0NTYgMTAuNDk0NiAwLjI0MDU3OCAxMC4yNTkgMC43NDY5QzguODU5MTMgMy43NTYwOCA0Ljc0MjQ3IDE0LjQxMTYgMTAuMjQwMyAyNS45OTMxQzkuNTgxNjUgMjUuODg2NCA4Ljg4NzUxIDI1LjgzMDMgOC4xNzAyMiAyNS44MzAzQzMuOTE4MzkgMjUuODMwMyAwIDI3Ljc5ODUgMCAzMC42NDA4QzAgMzMuNDgzIDMuOTE4MzkgMzUuMjI3MiA4LjE3MDIyIDM1LjIyNzJDOC43MTEyNyAzNS4yMjcyIDkuMjM5MjUgMzUuMTk4OCA5Ljc0ODkzIDM1LjE0MzVDOS40MzYwMiAzNS4yNjY0IDkuMTIyNzUgMzUuNDA3NSA4LjgxMTcxIDM1LjU2NzdDNS42OTM4OCAzNy4xNzA3IDMuOTY4OCA0MC4wMzEyIDQuOTU5MDQgNDEuOTU2OEM1Ljk0ODkgNDMuODgyNCA5LjI3OTIgNDQuMTQ0MiAxMi4zOTcgNDIuNTQxMkMxMy4wNDY0IDQyLjIwNzQgMTMuNjM0OCA0MS44MTkgMTQuMTUxNiA0MS4zOTZDMTguMjYyNyA0NC40OTY3IDI0LjcyODMgNDUuOTgwOSAzMS45OTk4IDQ1Ljk4MDlDMzkuMjcxMyA0NS45ODA5IDQ1LjczNyA0NC40OTY3IDQ5Ljg0OCA0MS4zOTZDNTAuMzY0NCA0MS44MTkgNTAuOTUzMyA0Mi4yMDc0IDUxLjYwMjYgNDIuNTQxMkM1NC43MjA0IDQ0LjE0NDIgNTguMDUwMyA0My44ODI0IDU5LjA0MDYgNDEuOTU2OEM2MC4wMzA1IDQwLjAzMTIgNTguMzA1NyAzNy4xNzA3IDU1LjE4NzkgMzUuNTY3N0M1NC44NzYxIDM1LjQwNzUgNTQuNTYyMSAzNS4yNjY3IDU0LjI0ODUgMzUuMTQzNUM1NC43NTg5IDM1LjE5ODggNTUuMjg3NiAzNS4yMjc1IDU1LjgyOTQgMzUuMjI3NUM2MC4wODEyIDM1LjIyNzUgNjMuOTk5NiAzMy40ODM0IDYzLjk5OTYgMzAuNjQxMUw2NCAzMC42NDA4WiIgZmlsbD0id2hpdGUiLz4KICAgIDxwYXRoIGQ9Ik0yMi4yMjkzIDM2Ljc0NDNDMjYuNTkzNSAzNi43NDQzIDMwLjEzMTQgMzMuMjA2NCAzMC4xMzE0IDI4Ljg0MjJDMzAuMTMxNCAyNC40NzggMjYuNTkzNSAyMC45NDAxIDIyLjIyOTMgMjAuOTQwMUMxNy44NjUxIDIwLjk0MDEgMTQuMzI3MSAyNC40NzggMTQuMzI3MSAyOC44NDIyQzE0LjMyNzEgMzMuMjA2NCAxNy44NjUxIDM2Ljc0NDMgMjIuMjI5MyAzNi43NDQzWiIgZmlsbD0iIzAwRTI4RSIvPgogICAgPHBhdGggZD0iTTIyLjI2NTUgMzMuMTM2MUMyMy41MDIyIDMzLjEzNjEgMjQuNTA0NyAzMS4yODA1IDI0LjUwNDcgMjguOTkxNUMyNC41MDQ3IDI2LjcwMjQgMjMuNTAyMiAyNC44NDY4IDIyLjI2NTUgMjQuODQ2OEMyMS4wMjg4IDI0Ljg0NjggMjAuMDI2MiAyNi43MDI0IDIwLjAyNjIgMjguOTkxNUMyMC4wMjYyIDMxLjI4MDUgMjEuMDI4OCAzMy4xMzYxIDIyLjI2NTUgMzMuMTM2MVoiIGZpbGw9IiMzQzZFOEIiLz4KICAgIDxwYXRoIGQ9Ik0zMS45OTk4IDM3LjkxNTZDMzMuNDIzNyAzNy45MTU2IDM0LjU3ODEgMzcuMzQwOSAzNC41NzgxIDM2LjYzMTlDMzQuNTc4MSAzNS45MjI5IDMzLjQyMzcgMzUuMzQ4MSAzMS45OTk4IDM1LjM0ODFDMzAuNTc1OCAzNS4zNDgxIDI5LjQyMTUgMzUuOTIyOSAyOS40MjE1IDM2LjYzMTlDMjkuNDIxNSAzNy4zNDA5IDMwLjU3NTggMzcuOTE1NiAzMS45OTk4IDM3LjkxNTZaIiBmaWxsPSIjRkY5NDkzIi8+CiAgICA8cGF0aCBkPSJNNTQuMjUwMyAzNS4xMDU4QzU0Ljc2IDM1LjE2MTEgNTUuMjg3OSAzNS4xODk0IDU1LjgyOSAzNS4xODk0QzYwLjA4MDggMzUuMTg5NCA2My45OTkyIDMzLjQ0NTMgNjMuOTk5MiAzMC42MDNDNjMuOTk5MiAyNy43NjA4IDYwLjA4MDggMjUuNzkyNiA1NS44MjkgMjUuNzkyNkM1NS4xMTE3IDI1Ljc5MjYgNTQuNDE3NiAyNS44NDkgNTMuNzU4NSAyNS45NTU4QzU5LjI1NjcgMTQuMzc0MiA1NS4xMzk3IDMuNzE4NzIgNTMuNzQwMiAwLjcwOTU0NkM1My41MDQ2IDAuMjAzMjI1IDUyLjk1MDEgLTAuMDcwMDk5MSA1Mi40MDM4IDAuMDQ1NjUyOUM0MS4zMjMgMi4zOTY5MSAzNi43OTIzIDEwLjI3MDcgMzUuNDI5OCAxMy4zMDE1QzM0LjQ1NDEgMTMuMTU2NiAzMy40NTc5IDEzLjA3MTEgMzIuNDQ1MiAxMy4wNTE3QzMxLjI3NDMgMjAuMDMzIDI4Ljk2NTYgNDMuOTM2NSA1NC4zNDM2IDM1LjE0MzlDNTQuMzEyMyAzNS4xMzEyIDU0LjI4MTMgMzUuMTE4MSA1NC4yNDk5IDM1LjEwNThINTQuMjUwM1oiIGZpbGw9IiMyQjlERkYiLz4KICAgIDxwYXRoIGQ9Ik00MS43MzQyIDMzLjEzNjFDNDIuOTcwOSAzMy4xMzYxIDQzLjk3MzUgMzEuMjgwNSA0My45NzM1IDI4Ljk5MTVDNDMuOTczNSAyNi43MDI0IDQyLjk3MDkgMjQuODQ2OCA0MS43MzQyIDI0Ljg0NjhDNDAuNDk3NSAyNC44NDY4IDM5LjQ5NSAyNi43MDI0IDM5LjQ5NSAyOC45OTE1QzM5LjQ5NSAzMS4yODA1IDQwLjQ5NzUgMzMuMTM2MSA0MS43MzQyIDMzLjEzNjFaIiBmaWxsPSIjM0M2RThCIi8+CiAgPC9nPgogIDxkZWZzPgogICAgPGNsaXBQYXRoIGlkPSJjbGlwMF8zMTBfMjI5Ij4KICAgICAgPHJlY3Qgd2lkdGg9IjY0IiBoZWlnaHQ9IjQ1Ljk2MTciIGZpbGw9IndoaXRlIiB0cmFuc2Zvcm09InRyYW5zbGF0ZSgwIDAuMDE5MTY1KSIvPgogICAgPC9jbGlwUGF0aD4KICA8L2RlZnM+Cjwvc3ZnPgo=';
public function __construct( $prefix, $mainfile, $domain, $isPro = false, $disableReview = false, $freeOnly = false ) {
// ALWAYS set instance properties first - these are needed regardless of when setup runs
$this->prefix = $prefix;
$this->mainfile = $mainfile;
$this->domain = $domain;
$this->isPro = $isPro;
$this->disableReview = $disableReview;
if ( is_admin() ) {
// Skip AJAX and REST requests to avoid unnecessary processing
if ( MeowKit_WPMC_Helpers::is_asynchronous_request() ) {
return;
}
// Check if WordPress pluggable functions are available yet.
// These are defined in wp-includes/pluggable.php, which WordPress loads
// AFTER the 'plugins_loaded' hook but BEFORE the 'init' hook.
if ( !function_exists( 'current_user_can' ) || !function_exists( 'wp_get_current_user' ) ) {
// Functions don't exist yet - defer admin setup until 'init' hook
// This is NORMAL behavior when plugins instantiate on 'plugins_loaded'
$this->defer_admin_setup();
// Continue to rest of constructor (filters, license checks, etc.)
} else {
// Functions already exist - safe to run admin setup immediately
// This happens when plugins instantiate on 'init' or later
$this->run_admin_setup();
}
// License-related admin notices (doesn't require pluggable functions)
$license = get_option( $this->prefix . '_license', '' );
if ( !empty( $license ) && !$this->isPro ) {
add_action( 'admin_notices', [ $this, 'admin_notices_licensed_free' ] );
}
}
// ALWAYS register these filters (they work at any time)
add_filter( 'plugin_row_meta', [ $this, 'custom_plugin_row_meta' ], 10, 2 );
add_filter( 'edd_sl_api_request_verify_ssl', [ $this, 'request_verify_ssl' ], 10, 0 );
}
/**
* Defer admin setup until WordPress 'init' hook.
*
* This method stores the current instance and registers a one-time
* 'init' hook callback that will process all deferred instances.
*
* Why defer? Because we need current_user_can() to check permissions,
* and that function doesn't exist until after 'plugins_loaded'.
*/
private function defer_admin_setup() {
// Add this instance to the queue for processing on 'init'
self::$deferred_instances[] = $this;
// Register the 'init' hook only once (for the first deferred instance)
if ( count( self::$deferred_instances ) === 1 ) {
add_action( 'init', array( __CLASS__, 'process_deferred_instances' ) );
}
}
/**
* Static callback for 'init' hook - processes all deferred instances.
*
* By the time 'init' fires, WordPress has loaded pluggable.php and
* current_user_can() is guaranteed to exist. We process all instances
* that were created during 'plugins_loaded' or earlier.
*
* This is called as a static method because it processes multiple instances.
*/
public static function process_deferred_instances() {
// Belt-and-suspenders check: pluggable functions should ALWAYS exist by 'init'
// If they somehow don't, log a warning and bail (this should never happen)
if ( !function_exists( 'current_user_can' ) || !function_exists( 'wp_get_current_user' ) ) {
trigger_error(
'MeowKit_WPMC_Admin: Pluggable functions still unavailable on init hook. ' .
'This should never happen and indicates a serious WordPress core issue.',
E_USER_WARNING
);
return;
}
// Process each deferred instance's admin setup
foreach ( self::$deferred_instances as $instance ) {
$instance->run_admin_setup();
}
// Clear the array to free memory (we won't need these references anymore)
self::$deferred_instances = array();
}
/**
* Run admin setup - both shared (once) and per-instance (each plugin).
*
* SHARED SETUP (once for all plugins):
* - Issues detection
* - Meow Apps menu creation
* - Admin footer customization
*
* PER-INSTANCE SETUP (once per plugin):
* - Ratings system
* - News system
*
* This method is called either immediately (if pluggable functions exist)
* or deferred until 'init' (if they don't). Either way, it's safe to call
* current_user_can() here.
*/
private function run_admin_setup() {
// SHARED SETUP: Only run once for all Meow Apps plugins
if ( !MeowKit_WPMC_Admin::$loaded ) {
// Check for potential issues with WordPress install, other plugins, etc.
new MeowKit_WPMC_Issues( $this->prefix, $this->mainfile, $this->domain );
// Create the unified Meow Apps menu (priority 5 to ensure early creation)
add_action( 'admin_menu', [ $this, 'admin_menu_start' ], 5 );
// Customize admin footer on Meow Apps pages
$page = isset( $_GET['page'] ) ? sanitize_text_field( $_GET['page'] ) : null;
if ( $page === 'meowapps-main-menu' ) {
add_filter( 'admin_footer_text', [ $this, 'admin_footer_text' ], 100000, 1 );
}
// Promote AI Engine on the WordPress 7 Connectors page when AI Engine
// itself isn't installed. When AI Engine is active, its own banner
// takes over — so this path only runs on "bare" Meow Apps installs.
add_action( 'admin_footer', [ $this, 'maybe_render_wpai_promo' ] );
MeowKit_WPMC_Admin::$loaded = true;
}
// PER-INSTANCE SETUP: Run for each plugin that uses this library
// Only admins get ratings prompts and news
if ( $this->is_user_admin() ) {
if ( !$this->disableReview ) {
new MeowKit_WPMC_Ratings( $this->prefix, $this->mainfile, $this->domain );
}
new MeowKit_WPMC_News( $this->domain );
}
}
/**
* Check if current user is a site administrator.
*
* This method is only called from run_admin_setup(), which guarantees
* that pluggable functions exist. No error logging needed - if the
* functions don't exist, we simply return false as a defensive fallback.
*
* @return bool True if user can manage options, false otherwise
*/
public function is_user_admin() {
// Defensive check (should never fail if called from run_admin_setup)
if ( !function_exists( 'current_user_can' ) || !function_exists( 'wp_get_current_user' ) ) {
return false;
}
return current_user_can( 'manage_options' );
}
public function custom_plugin_row_meta( $links, $file ) {
$path = pathinfo( $file );
$pathName = basename( $path['dirname'] );
$thisPath = pathinfo( $this->mainfile );
$thisPathName = basename( $thisPath['dirname'] );
$isActive = is_plugin_active( $file );
if ( !$isActive ) {
return $links;
}
$isIssue = $this->isPro && !$this->is_registered();
if ( strpos( $pathName, $thisPathName ) !== false ) {
// In network admin, handle differently (no settings page available)
if ( is_network_admin() ) {
if ( $this->isPro && !$this->is_registered() ) {
// Show "Register License" link for unregistered Pro plugins
$new_links = [
'license' => sprintf(
'<a href="#" class="meowapps-network-license-link" data-prefix="%s" data-plugin="%s" style="color: #d63638;">%s</a>',
esc_attr( $this->prefix ),
esc_attr( $this->nice_name_from_file( $this->mainfile ) ),
__( 'Register License', $this->domain )
),
];
// Track this plugin for the modal
self::$network_license_plugins[ $this->prefix ] = $this->nice_name_from_file( $this->mainfile );
// Add modal output hook (only once)
if ( !self::$network_license_modal_added ) {
add_action( 'admin_footer', [ __CLASS__, 'output_network_license_modal' ] );
self::$network_license_modal_added = true;
}
}
elseif ( $this->isPro && $this->is_registered() ) {
// Pro plugin is registered
$new_links = [
'license' => '<span style="color: #a75bd6;">' . __( 'Pro Version', $this->domain ) . '</span>',
];
}
else {
// Free plugin
$new_links = [
'license' => sprintf( '<span>' . __( '<a target="_blank" href="https://meowapps.com">Get the <u>Pro Version</u></a>', $this->domain ), $this->prefix ) . '</span>',
];
}
}
else {
// Regular admin - show settings and license status
$new_links = [
'settings' =>
sprintf( __( '<a href="admin.php?page=%s_settings">Settings</a>', $this->domain ), $this->prefix ),
'license' =>
$this->is_registered() ?
( '<span style="color: #a75bd6;">' . __( 'Pro Version', $this->domain ) . '</span>' ) :
( $isIssue ? ( sprintf( '<span style="color: #ff3434;">' . __( 'License Issue', $this->domain ), $this->prefix ) . '</span>' ) : ( sprintf( '<span>' . __( '<a target="_blank" href="https://meowapps.com">Get the <u>Pro Version</u></a>', $this->domain ), $this->prefix ) . '</span>' ) ),
];
}
$links = array_merge( $new_links, $links );
}
return $links;
}
/**
* Output the network license registration modal.
* Called via admin_footer hook in network admin.
*/
public static function output_network_license_modal() {
$rest_url = esc_url( rest_url() );
$nonce = wp_create_nonce( 'wp_rest' );
?>
<div id="meowapps-network-license-modal" style="display:none; position:fixed; top:0; left:0; width:100%; height:100%; background:rgba(0,0,0,0.6); z-index:100000; align-items:center; justify-content:center;">
<div style="background:#fff; padding:24px; border-radius:8px; max-width:450px; width:90%; box-shadow:0 4px 20px rgba(0,0,0,0.3);">
<h2 style="margin:0 0 8px 0; font-size:18px;">Register License</h2>
<p id="meowapps-license-plugin-name" style="margin:0 0 16px 0; color:#666;"></p>
<input type="text" id="meowapps-license-key-input" placeholder="Enter your license key" style="width:100%; padding:10px; font-size:14px; border:1px solid #8c8f94; border-radius:4px; box-sizing:border-box;" />
<p id="meowapps-license-message" style="margin:12px 0 0 0; padding:10px; border-radius:4px; display:none;"></p>
<div style="margin-top:16px; display:flex; gap:10px; justify-content:flex-end;">
<button type="button" id="meowapps-license-cancel" class="button">Cancel</button>
<button type="button" id="meowapps-license-submit" class="button button-primary">Validate & Register</button>
</div>
</div>
</div>
<script>
(function() {
var modal = document.getElementById('meowapps-network-license-modal');
var input = document.getElementById('meowapps-license-key-input');
var message = document.getElementById('meowapps-license-message');
var pluginName = document.getElementById('meowapps-license-plugin-name');
var submitBtn = document.getElementById('meowapps-license-submit');
var cancelBtn = document.getElementById('meowapps-license-cancel');
var currentPrefix = '';
function showMessage(text, isError) {
message.textContent = text;
message.style.display = 'block';
message.style.background = isError ? '#fcf0f1' : '#edfaef';
message.style.color = isError ? '#d63638' : '#1e7e34';
message.style.border = '1px solid ' + (isError ? '#d63638' : '#1e7e34');
}
function hideMessage() {
message.style.display = 'none';
}
function openModal(prefix, plugin) {
currentPrefix = prefix;
pluginName.textContent = plugin;
input.value = '';
hideMessage();
submitBtn.disabled = false;
submitBtn.textContent = 'Validate & Register';
modal.style.display = 'flex';
input.focus();
}
function closeModal() {
modal.style.display = 'none';
currentPrefix = '';
}
// Handle click on "Register License" links
document.addEventListener('click', function(e) {
if (e.target.classList.contains('meowapps-network-license-link')) {
e.preventDefault();
var prefix = e.target.getAttribute('data-prefix');
var plugin = e.target.getAttribute('data-plugin');
openModal(prefix, plugin);
}
});
// Close modal on cancel or clicking outside
cancelBtn.addEventListener('click', closeModal);
modal.addEventListener('click', function(e) {
if (e.target === modal) closeModal();
});
// Handle escape key
document.addEventListener('keydown', function(e) {
if (e.key === 'Escape' && modal.style.display === 'flex') {
closeModal();
}
});
// Handle enter key in input
input.addEventListener('keydown', function(e) {
if (e.key === 'Enter') {
submitBtn.click();
}
});
// Submit license
submitBtn.addEventListener('click', function() {
var licenseKey = input.value.trim();
if (!licenseKey) {
showMessage('Please enter a license key.', true);
return;
}
submitBtn.disabled = true;
submitBtn.textContent = 'Validating...';
hideMessage();
var restUrl = '<?php echo $rest_url; ?>meow-licenser/' + currentPrefix + '/v1/set_license/';
fetch(restUrl, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-WP-Nonce': '<?php echo $nonce; ?>'
},
body: JSON.stringify({ serialKey: licenseKey })
})
.then(function(response) { return response.json(); })
.then(function(data) {
if (data.success && data.data && !data.data.issue) {
showMessage('License registered successfully! Reloading...', false);
setTimeout(function() { location.reload(); }, 1500);
} else {
var errorMsg = 'License validation failed.';
if (data.data && data.data.issue) {
errorMsg = 'License issue: ' + data.data.issue;
}
showMessage(errorMsg, true);
submitBtn.disabled = false;
submitBtn.textContent = 'Validate & Register';
}
})
.catch(function(error) {
showMessage('Error: ' + error.message, true);
submitBtn.disabled = false;
submitBtn.textContent = 'Validate & Register';
});
});
})();
</script>
<?php
}
public function request_verify_ssl() {
return get_option( 'force_sslverify', false );
}
public function nice_name_from_file( $file ) {
$info = pathinfo( $file );
if ( !empty( $info ) ) {
if ( $info['filename'] == 'wplr-sync' ) {
return 'WP/LR Sync';
}
$info['filename'] = str_replace( '-', ' ', $info['filename'] );
$file = ucwords( $info['filename'] );
}
return $file;
}
public function admin_notices_licensed_free() {
if ( isset( $_POST[$this->prefix . '_reset_sub'] ) ) {
delete_option( $this->prefix . '_pro_serial' );
delete_option( $this->prefix . '_license' );
return;
}
$html = '<div class="notice notice-error">';
$html .= sprintf(
__( '<p>It looks like you are using the free version of the plugin (<b>%s</b>) but a license for the Pro version was also found. The Pro version might have been replaced by the Free version during an update (might be caused by a temporarily issue). If it is the case, <b>please download it again</b> from the <a target="_blank" href="https://meowapps.com">Meow Store</a>. If you wish to continue using the free version and clear this message, click on this button.', $this->domain ),
$this->nice_name_from_file( $this->mainfile )
);
$html .= '<p>
<form method="post" action="">
<input type="hidden" name="' . $this->prefix . '_reset_sub" value="true">
<input type="submit" name="submit" id="submit" class="button" value="'
. __( 'Remove the license', $this->domain ) . '">
</form>
</p>';
$html .= '</div>';
wp_kses_post( $html );
}
public function admin_menu_start() {
// Hide the admin if user doesn't like Meow much
if ( get_option( 'meowapps_hide_meowapps', false ) ) {
register_setting( 'general', 'meowapps_hide_meowapps' );
add_settings_field( 'meowapps_hide_ads', 'Meow Apps Menu', [ $this, 'meowapps_hide_dashboard_callback' ], 'general' );
return;
}
// Create standard menu if it does not already exist
global $submenu;
if ( !isset( $submenu[ 'meowapps-main-menu' ] ) ) {
add_menu_page(
'Meow Apps',
'<img alt="Meow Apps" style="width: 21px; margin-left: -28px; position: absolute; margin-top: 2px;" src="' . MeowKit_WPMC_Admin::$logo . '" />Meow Apps',
'manage_options',
'meowapps-main-menu',
[ $this, 'admin_meow_apps' ],
'',
82
);
add_submenu_page(
'meowapps-main-menu',
__( 'Dashboard', $this->domain ),
__( 'Dashboard', $this->domain ),
'manage_options',
'meowapps-main-menu',
[ $this, 'admin_meow_apps' ]
);
}
// Add CSS to hide the default icon
add_action( 'admin_head', function () {
echo '<style>
#toplevel_page_meowapps-main-menu .wp-menu-image {
display: none;
}
</style>';
} );
}
public function meowapps_hide_dashboard_callback() {
$html = '<input type="checkbox" id="meowapps_hide_meowapps" name="meowapps_hide_meowapps" value="1" ' .
checked( 1, get_option( 'meowapps_hide_meowapps' ), false ) . '/>';
$html .= __( '<label>Hide <b>Meow Apps</b> Menu</label><br /><small>Hide Meow Apps menu and all its components, for a cleaner admin. This option will be reset if a new Meow Apps plugin is installed.<br /><b>Once activated, an option will be added in your General settings to display it again.</b></small>', $this->domain );
echo MeowKit_WPMC_Helpers::wp_kses( $html );
}
public function is_registered() {
$is_registered = apply_filters( $this->prefix . '_meowapps_is_registered', false, $this->prefix );
return $is_registered;
}
public function get_phpinfo() {
if ( !$this->is_user_admin() || !function_exists( 'phpinfo' ) ) {
return;
}
ob_start();
// phpcs:disable WordPress.PHP.DevelopmentFunctions
phpinfo( INFO_GENERAL | INFO_CONFIGURATION | INFO_MODULES );
// phpcs:enable
$html = ob_get_contents();
ob_end_clean();
$html = preg_replace( '%^.*<body>(.*)</body>.*$%ms', '$1', $html );
return $html;
}
public function admin_meow_apps() {
$html = "<div id='meow-common-dashboard'></div>";
$html .= "<div style='height: 0; width: 0; overflow: hidden;' id='meow-common-phpinfo'>";
$html .= $this->get_phpinfo();
$html .= '</div>';
$html = preg_replace( "/<img[^>]+\>/i", '', $html );
echo wp_kses_post( $html );
}
public function admin_footer_text( $current ) {
return sprintf(
// translators: %1$s is the version of the interface; %2$s is a file path.
__( 'Thanks for using <a href="https://meowapps.com">Meow Apps</a>! This is the Meow Admin %1$s <br /><i>Loaded from %2$s </i>', $this->domain ),
MeowKit_WPMC_Admin::$version,
__FILE__
);
}
/**
* Renders a promo banner on WordPress 7's Connectors page when AI Engine
* isn't installed. Kept self-contained so the common library stays simple:
* no new file, no REST endpoint, dismissal persists in localStorage.
*/
public function maybe_render_wpai_promo() {
// WordPress 7+ only.
if ( ! class_exists( 'WP_Connector_Registry' ) ) {
return;
}
// If AI Engine is installed, its own Connectors banner takes over.
if ( class_exists( 'Meow_MWAI_Core' ) ) {
return;
}
// Another Meow Apps plugin's common copy may have rendered already.
if ( defined( 'MEOWAPPS_WPAI_PROMO_RENDERED' ) ) {
return;
}
// Gate on the Connectors screen. The hook suffix differs between the
// direct file (`options-connectors.php`) and the menu-page variant.
$screen = function_exists( 'get_current_screen' ) ? get_current_screen() : null;
$id = $screen ? $screen->id : ( isset( $GLOBALS['hook_suffix'] ) ? $GLOBALS['hook_suffix'] : '' );
$targets = array( 'options-connectors', 'options-connectors.php', 'settings_page_options-connectors-wp-admin' );
if ( ! in_array( $id, $targets, true ) ) {
return;
}
define( 'MEOWAPPS_WPAI_PROMO_RENDERED', true );
// Install button → WordPress's own plugin-install search page, pre-
// filtered for AI Engine. One click from there to install. This is the
// most reliable path: no custom nonces, native progress UI, native
// filesystem credential prompt if needed.
$can_install = current_user_can( 'install_plugins' );
$install_url = $can_install
? self_admin_url( 'plugin-install.php?tab=search&type=term&s=AI+Engine' )
: 'https://wordpress.org/plugins/ai-engine/';
$wporg_url = 'https://wordpress.org/plugins/ai-engine/';
$learn_url = 'https://meowapps.com/wordpress-7-ai-engine-gateway/';
// Title is split so "AI Engine" can carry an anchor to wp.org. Keeping
// the pieces as data (not one HTML string) avoids escaping surprises and
// keeps the translation unit stable.
$payload = array(
'titleBefore' => __( 'Highly recommended: Let ', 'meowapps' ),
'titleLink' => __( 'AI Engine', 'meowapps' ),
'titleAfter' => __( ' handle your connections.', 'meowapps' ),
'sub' => __( 'One plugin for every provider. Keep your AI setup clean and consistent: monitor your API costs in one place, log every single request, and avoid the mess of juggling separate plugins for each AI model.', 'meowapps' ),
'install' => __( 'Install AI Engine', 'meowapps' ),
'learn' => __( 'Learn more', 'meowapps' ),
'dismiss' => __( 'Dismiss', 'meowapps' ),
'installUrl' => $install_url,
'wporgUrl' => $wporg_url,
'learnUrl' => $learn_url,
);
?>
<style>
.meowapps-wpai-promo {
margin: 0 0 16px; padding: 14px 18px;
display: flex; align-items: flex-start; gap: 14px;
background: #f0f4ff; border: 1px solid #d6deff; border-radius: 4px;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
font-size: 13px; line-height: 1.45; color: #1e1e1e;
}
.meowapps-wpai-promo-icon {
width: 32px; height: 32px; border-radius: 50%;
background: #2f5fff; color: #fff;
display: flex; align-items: center; justify-content: center;
flex-shrink: 0;
margin-top: 1px;
}
.meowapps-wpai-promo-icon svg { width: 18px; height: 18px; display: block; }
.meowapps-wpai-promo-body {
flex: 1; min-width: 0;
display: flex; flex-direction: column; gap: 10px;
}
.meowapps-wpai-promo-text strong { font-weight: 700; display: block; margin-bottom: 3px; }
.meowapps-wpai-promo-text span { color: #50575e; }
.meowapps-wpai-promo-titlelink {
color: #2f5fff; text-decoration: none; border-bottom: 1px dashed #2f5fff;
}
.meowapps-wpai-promo-titlelink:hover { color: #2448cc; border-bottom-color: #2448cc; }
.meowapps-wpai-promo-actions {
display: flex; gap: 8px; flex-wrap: wrap; align-items: center;
}
.meowapps-wpai-promo-btn {
appearance: none; border: 1px solid transparent; border-radius: 4px;
padding: 7px 16px; font: inherit; font-size: 12.5px; font-weight: 600;
cursor: pointer; text-decoration: none;
transition: background 0.12s ease, border-color 0.12s ease;
}
.meowapps-wpai-promo-btn-primary { background: #2f5fff; color: #fff; }
.meowapps-wpai-promo-btn-primary:hover { background: #2448cc; color: #fff; }
.meowapps-wpai-promo-btn-secondary { background: #7c3aed; color: #fff; }
.meowapps-wpai-promo-btn-secondary:hover { background: #6527c9; color: #fff; }
.meowapps-wpai-promo-btn-dismiss {
margin-left: auto;
background: transparent; color: #6b7280;
font-weight: 500; padding: 7px 10px;
}
.meowapps-wpai-promo-btn-dismiss:hover { color: #1e1e1e; background: rgba(0,0,0,0.04); }
</style>
<script>
(function () {
var D = <?php echo wp_json_encode( $payload ); ?>;
try { if (localStorage.getItem('meowapps-wpai-promo-dismissed') === '1') return; } catch (e) {}
function build() {
var host = document.createElement('div');
host.className = 'meowapps-wpai-promo';
host.setAttribute('role', 'status');
var iconSvg = '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><path d="M12 5v14M5 12h14"/></svg>';
host.innerHTML = [
'<span class="meowapps-wpai-promo-icon">', iconSvg, '</span>',
'<div class="meowapps-wpai-promo-body">',
'<div class="meowapps-wpai-promo-text">',
'<strong></strong> <span class="meowapps-wpai-promo-sub"></span>',
'</div>',
'<div class="meowapps-wpai-promo-actions"></div>',
'</div>'
].join('');
// Title carries an anchor on "AI Engine" → wp.org plugin page.
var strong = host.querySelector('strong');
strong.appendChild(document.createTextNode(D.titleBefore));
var tLink = document.createElement('a');
tLink.href = D.wporgUrl;
tLink.target = '_blank';
tLink.rel = 'noopener noreferrer';
tLink.className = 'meowapps-wpai-promo-titlelink';
tLink.textContent = D.titleLink;
strong.appendChild(tLink);
strong.appendChild(document.createTextNode(D.titleAfter));
host.querySelector('.meowapps-wpai-promo-sub').textContent = D.sub;
var actions = host.querySelector('.meowapps-wpai-promo-actions');
function link(label, cls, href, newTab) {
var a = document.createElement('a');
a.className = 'meowapps-wpai-promo-btn ' + cls;
a.textContent = label;
a.href = href;
if (newTab) { a.target = '_blank'; a.rel = 'noopener noreferrer'; }
actions.appendChild(a);
return a;
}
// Install → WordPress's plugin-install search page, pre-filtered.
// User lands on a familiar screen with AI Engine as the top hit and
// can install with the native WordPress UX.
link(D.install, 'meowapps-wpai-promo-btn-primary', D.installUrl, false);
link(D.learn, 'meowapps-wpai-promo-btn-secondary', D.learnUrl, true);
var d = document.createElement('button');
d.type = 'button';
d.className = 'meowapps-wpai-promo-btn meowapps-wpai-promo-btn-dismiss';
d.textContent = D.dismiss;
d.addEventListener('click', function () {
try { localStorage.setItem('meowapps-wpai-promo-dismissed', '1'); } catch (e) {}
if (host.parentNode) host.parentNode.removeChild(host);
});
actions.appendChild(d);
return host;
}
function ensure() {
if (document.querySelector('.meowapps-wpai-promo')) return;
var page = document.querySelector('.connectors-page');
if (page) { page.insertBefore(build(), page.firstChild); return; }
var header = document.querySelector('.boot-layout__stage header');
if (header && header.parentNode) {
header.parentNode.insertBefore(build(), header.nextSibling);
}
}
var tries = 0;
var iv = setInterval(function () {
ensure();
if (document.querySelector('.meowapps-wpai-promo') || ++tries > 40) clearInterval(iv);
}, 120);
var app = document.getElementById('options-connectors-wp-admin-app')
|| document.getElementById('options-connectors-app');
if (app && 'MutationObserver' in window) {
new MutationObserver(ensure).observe(app, { childList: true, subtree: true });
}
})();
</script>
<?php
}
}
}