HEX
Server: Apache/2.4.65 (Unix) OpenSSL/1.1.1k
System: Linux vps109042.inmotionhosting.com 4.18.0 #1 SMP Mon Sep 30 15:36:27 MSK 2024 x86_64
User: cisa (1010)
PHP: 8.2.30
Disabled: NONE
Upload Files
File: /home/cisa/public_html/wp-content/plugins/charitable/charitable.php
<?php
/**
 * Plugin Name: Charitable
 * Plugin URI: https://www.wpcharitable.com
 * Description: The best WordPress donation plugin. Fundraising with recurring donations, and powerful features to help you raise more money online.
 * Version: 1.8.9.7
 * Author: Charitable Donations & Fundraising Team
 * Author URI: https://wpcharitable.com
 * Requires at least: 5.0
 * Stable tag: 1.8.9.7
 * License: GPLv2 or later
 * License URI: https://www.gnu.org/licenses/old-licenses/gpl-2.0.html
 *
 * Text Domain: charitable
 * Domain Path: /i18n/languages/
 *
 * @package   Charitable
 * @author    David Bisset
 * @copyright Copyright (c) 2026, WP Charitable LLC
 * @license   http://opensource.org/licenses/gpl-2.0.php GNU Public License
 */

// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

if ( ! class_exists( 'Charitable' ) ) :

	/**
	 * Main Charitable class.
	 */
	class Charitable {

		/** The product name. */
		const NAME = 'Charitable';

		/** The plugin author name. */
		const AUTHOR = 'WP Charitable';

		/* Plugin version. */
		const VERSION = '1.8.9.7';

		/* Version of database schema. */
		const DB_VERSION = '20180522';

		/* Campaign post type. */
		const CAMPAIGN_POST_TYPE = 'campaign';

		/* Donation post type. */
		const DONATION_POST_TYPE = 'donation';

		/**
		 * Single instance of this class.
		 *
		 * @since 1.0.0
		 *
		 * @var   Charitable
		 */
		private static $instance = null;

		/**
		 * The absolute path to this plugin's directory.
		 *
		 * @since 1.0.0
		 *
		 * @var   string
		 */
		private $directory_path;

		/**
		 * The URL of this plugin's directory.
		 *
		 * @since 1.0.0
		 *
		 * @var   string
		 */
		private $directory_url;

		/**
		 * Directory path for the includes folder of the plugin.
		 *
		 * @since 1.0.0
		 *
		 * @var   string
		 */
		private $includes_path;

		/**
		 * Store of registered objects.
		 *
		 * @since 1.0.0
		 * @since 1.5.0 Changed to Charitable_Registry object. Previously it was an array.
		 *
		 * @var   Charitable_Registry
		 */
		private $registry;

		/**
		 * Classmap.
		 *
		 * @since 1.5.0
		 *
		 * @var   array
		 */
		private $classmap;

		/**
		 * Plugin.
		 *
		 * @since 1.8.0
		 *
		 * @var   array
		 */
		private $plugin;

		/**
		 * Create class instance.
		 *
		 * @since 1.0.0
		 */
		public function __construct() {
			$this->directory_path = plugin_dir_path( __FILE__ );
			$this->directory_url  = plugin_dir_url( __FILE__ );
			$this->includes_path  = $this->directory_path . 'includes/';

			define( 'CHARITABLE_DIRECTORY_PATH', plugin_dir_path( __FILE__ ) );
			define( 'CHARITABLE_PLUGIN_BASENAME', plugin_basename( __FILE__ ) );
			define( 'CHARITABLE_MARKETING_URL', 'https://wpcharitable.com/' );

			define( 'CHARITABLE_BUILDER_SHOW_LEGACY_EDIT_LINKS', true ); // 1.8.0 specific.

			spl_autoload_register( array( $this, 'autoloader' ) );

			$this->load_dependencies();

			register_activation_hook( __FILE__, array( $this, 'activate' ) );
			register_deactivation_hook( __FILE__, array( $this, 'deactivate' ) );

			add_filter( 'plugin_action_links_' . CHARITABLE_PLUGIN_BASENAME, array( $this, 'plugin_action_links' ), 99, 2 );
			add_action( 'plugins_loaded', array( $this, 'start' ), 3 );

			add_action( 'plugins_loaded', array( $this, 'check_for_version_conflicts' ), 2 );

			// Run upgrade-server callback before any redirect (e.g. login) can run. Request from connect-callback.php.
			if ( defined( 'CHARITABLE_CONNECT_CALLBACK' ) && CHARITABLE_CONNECT_CALLBACK ) {
				add_action( 'init', array( $this, 'run_connect_callback' ), -9999 );
			}

			// Initialize error handler early.
			add_action(
				'plugins_loaded',
				function () {
					require_once $this->get_path( 'includes' ) . 'class-charitable-error-handler.php';
					$error_handler = new Charitable_Error_Handler();
					$error_handler->init();
				},
				1
			);
		}

		/**
		 * Returns the original instance of this class.
		 *
		 * @since  1.0.0
		 *
		 * @return Charitable
		 */
		public static function get_instance() {
			return self::$instance;
		}

		/**
		 * Run the startup sequence.
		 *
		 * This is only ever executed once.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function start() {
			/* If we've already started (i.e. run this function once before), do not pass go. */
			if ( $this->started() ) {
				return;
			}

			/* Set static instance. */
			self::$instance = $this;

			/* Set up the registry and instantiate objects we need straight away. */
			$this->registry();

			$this->setup_licensing();

			$this->maybe_start_ajax();

			$this->attach_hooks_and_filters();

			$this->maybe_start_admin();

			$this->maybe_start_public();

			Charitable_Addons::load( $this );
		}

		/**
		 * Include necessary files.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		private function load_dependencies() {
			$includes_path = $this->get_path( 'includes' );

			/* Load files with hooks & functions. Classes are autoloaded. */
			require_once $includes_path . 'charitable-core-functions.php';
			require_once $includes_path . 'api/charitable-api-functions.php';
			require_once $includes_path . 'campaigns/charitable-campaign-functions.php';
			require_once $includes_path . 'campaigns/charitable-campaign-hooks.php';
			require_once $includes_path . 'compat/charitable-compat-functions.php';
			require_once $includes_path . 'currency/charitable-currency-functions.php';
			require_once $includes_path . 'deprecated/charitable-deprecated-functions.php';
			require_once $includes_path . 'donations/charitable-donation-hooks.php';
			require_once $includes_path . 'donations/charitable-donation-functions.php';
			require_once $includes_path . 'emails/charitable-email-hooks.php';
			require_once $includes_path . 'endpoints/charitable-endpoints-functions.php';
			require_once $includes_path . 'privacy/charitable-privacy-functions.php';
			require_once $includes_path . 'public/charitable-template-helpers.php';
			require_once $includes_path . 'shortcodes/charitable-shortcodes-hooks.php';
			require_once $includes_path . 'upgrades/charitable-upgrade-hooks.php';
			require_once $includes_path . 'users/charitable-user-functions.php';
			require_once $includes_path . 'user-management/charitable-user-management-hooks.php';
			require_once $includes_path . 'utilities/charitable-utility-functions.php';
			require_once $includes_path . 'elementor/charitable-elementor-hooks.php';

			// Load security hooks on both admin and frontend (CAPTCHA needs to work on frontend).
			require_once $this->get_path( 'includes', true ) . 'admin/security/charitable-security-hooks.php';

			/* Load vendor/autoload.php - our modified platform_check.php and autoload_real.php will handle conditional loading */
			require_once $this->get_path( 'directory' ) . 'vendor/autoload.php';

			/* Square Compatibility Wrapper - always load this regardless of PHP version */
			require_once $includes_path . 'square/class-charitable-square-compatibility.php';

			/* Initialize Square conditionally - will check for compatibility internally */
			Charitable_Square_Compatibility::init();

			if ( $this->load_core_stripe() ) :

				/* Interfaces */
				require_once $includes_path . 'stripe/interfaces/interface-charitable-stripe-gateway-processor.php';

				/* Abstract classes */
				require_once $includes_path . 'stripe/abstracts/class-charitable-stripe-gateway-processor.php';

				/* Classes */
				require_once $includes_path . 'stripe/donations/class-charitable-stripe-donation-log.php';
				require_once $includes_path . 'stripe/donations/class-charitable-stripe-recurring-donation-log.php';
				require_once $includes_path . 'stripe/i18n/class-charitable-stripe-i18n.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-connected-customer.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-customer.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-gateway-processor-payment-intents.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-payment-intent.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-plan.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-product.php';

				require_once $includes_path . 'stripe/connect/charitable-stripe-connect-functions.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-gateway-processor-checkout.php';

				/* Functions & Hooks */
				require_once $includes_path . 'stripe/charitable-stripe-core-functions.php';
				require_once $includes_path . 'stripe/donations/charitable-stripe-donation-functions.php';
				require_once $includes_path . 'stripe/gateway/charitable-stripe-gateway-hooks.php';

				/* webhooks */
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-webhook-api.php';
				require_once $includes_path . 'stripe/gateway/class-charitable-stripe-webhook-processor.php';

			endif;
		}

		/**
		 * Load the template functions after theme is loaded.
		 *
		 * This gives themes time to override the functions.
		 *
		 * @since  1.6.10
		 *
		 * @return void
		 */
		public function load_template_files() {
			$includes_path = $this->get_path( 'includes' );

			require_once $includes_path . 'public/charitable-template-functions.php';
			require_once $includes_path . 'public/charitable-template-hooks.php';
		}

		/**
		 * Dynamically loads the class attempting to be instantiated elsewhere in the
		 * plugin by looking at the $class_name parameter being passed as an argument.
		 *
		 * @since  1.5.0
		 *
		 * @param  string $class_name The fully-qualified name of the class to load.
		 * @return boolean
		 */
		public function autoloader( $class_name ) {
			/* If the specified $class_name already exists, bail. */
			if ( class_exists( $class_name ) ) {
				return false;
			}

			/* If the specified $class_name does not include our namespace, duck out. */
			if ( false === strpos( $class_name, 'Charitable_' ) ) {
				return false;
			}

			/* Autogenerated class map. */
			if ( ! isset( $this->classmap ) ) {
				$this->classmap = include 'includes/autoloader/charitable-class-map.php';

				$this->classmap['Charitable_Gateway_Stripe_AM'] = 'gateways/class-charitable-gateway-stripe-am.php';
				$this->classmap['Charitable_Gateway_Stripe']    = 'gateways/class-charitable-gateway-stripe-am.php';
			}

			$file_path = isset( $this->classmap[ $class_name ] ) ? $this->get_path( 'includes' ) . $this->classmap[ $class_name ] : false;

			if ( false !== $file_path && file_exists( $file_path ) && is_file( $file_path ) ) {
				require_once $file_path;
				return true;
			}
			return false;
		}

		/**
		 * Returns a registered class object.
		 *
		 * @since  1.5.0
		 *
		 * @return Charitable_Registry
		 */
		public function registry() {
			if ( ! isset( $this->registry ) ) {
				$this->registry = new Charitable_Registry();
				$this->registry->register_object( Charitable_Emails::get_instance() );
				$this->registry->register_object( Charitable_Request::get_instance() );
				$this->registry->register_object( Charitable_Gateways::get_instance() );
				$this->registry->register_object( Charitable_i18n::get_instance() );
				$this->registry->register_object( Charitable_Post_Types::get_instance() );
				$this->registry->register_object( Charitable_Cron::get_instance() );
				$this->registry->register_object( Charitable_Widgets::get_instance() );
				$this->registry->register_object( Charitable_Licenses::get_instance() );
				$this->registry->register_object( Charitable_User_Dashboard::get_instance() );
				$this->registry->register_object( Charitable_Locations::get_instance() );
				$this->registry->register_object( Charitable_Currency::get_instance() );
				$this->registry->register_object( Charitable_Notifications::get_instance() );

				$this->registry->register_object( new Charitable_Privacy() );
				$this->registry->register_object( new Charitable_Debugging() );
				$this->registry->register_object( new Charitable_Locale() );
			}

			return $this->registry;
		}

		/**
		 * Set up hook and filter callback functions.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		private function attach_hooks_and_filters() {
			add_action( 'wpmu_new_blog', array( $this, 'maybe_activate_charitable_on_new_site' ) );
			add_action( 'plugins_loaded', array( $this, 'charitable_install' ), 100 );
			add_action( 'plugins_loaded', array( $this, 'charitable_start' ), 100 );
			add_action( 'plugins_loaded', array( $this, 'endpoints' ), 100 );
			add_action( 'init', array( $this, 'donation_fields' ), 100 );
			add_action( 'init', array( $this, 'campaign_fields' ), 100 );
			add_action( 'plugins_loaded', array( $this, 'register_donormeta_table' ) );
			add_action( 'plugins_loaded', 'charitable_load_compat_functions' );
			add_action( 'setup_theme', array( 'Charitable_Customizer', 'start' ) );
			add_action( 'after_setup_theme', array( $this, 'load_template_files' ) );
			add_action( 'wp_enqueue_scripts', array( $this, 'maybe_start_qunit' ), 100 );
			add_action( 'rest_api_init', 'charitable_register_api_routes' );

			/**
			 * We do this on priority 20 so that any functionality that is loaded on init (such
			 * as addons) has a chance to run before the event.
			 */
			add_action( 'init', array( $this, 'do_charitable_actions' ), 20 );

			if ( $this->load_core_stripe() ) :

				/**
				 * Set up scripts & stylesheets & enqueue them when appropriate.
				 */
				add_action( 'charitable_form_after_fields', array( $this, 'maybe_setup_stripe_scripts_in_donation_form' ) );
				add_action( 'charitable_campaign_loop_after', array( $this, 'maybe_setup_stripe_scripts_in_campaign_loop' ) );

			endif;
		}


		/**
		 * Load Stripe JS, as well as our handling scripts.
		 *
		 * @since  1.4.0
		 *
		 * @return boolean
		 */
		public function enqueue_stripe_scripts() {
			if ( ! Charitable_Gateways::get_instance()->is_active_gateway( Charitable_Gateway_Stripe_AM::get_gateway_id() ) ) {
				return false;
			}

			wp_enqueue_script( 'charitable-stripe' );

			return true;
		}

		/**
		 * Load Stripe JS or Stripe Checkout, as well as our handling script.
		 *
		 * @uses   Charitable_Stripe::enqueue_scripts()
		 *
		 * @since  1.4.0
		 *
		 * @param  Charitable_Donation_Form $form The current form object.
		 * @return boolean
		 */
		public function maybe_setup_stripe_scripts_in_donation_form( $form ) {
			if ( ! is_a( $form, 'Charitable_Donation_Form' ) ) {
				return false;
			}

			if ( 'make_donation' !== $form->get_form_action() ) {
				return false;
			}

			return $this->enqueue_stripe_scripts();
		}

		/**
		 * Enqueue the Stripe JS/Checkout scripts after a campaign loop if modal donations are in use.
		 *
		 * @uses Charitable_Stripe::enqueue_scripts()
		 *
		 * @since  1.4.0
		 *
		 * @return boolean
		 */
		public function maybe_setup_stripe_scripts_in_campaign_loop() {
			if ( 'modal' !== charitable_get_option( 'donation_form_display', 'separate_page' ) ) {
				return false;
			}

			return $this->enqueue_stripe_scripts();
		}

		/**
		 * Checks whether we're in the admin area and if so, loads the admin-only functionality.
		 *
		 * @since  1.0.0
		 *
		 * @return Charitable_Admin|false
		 */
		private function maybe_start_admin() {
			if ( ! is_admin() ) {
				return false;
			}

			// load blocks/Gutenberg related.
			require_once $this->get_path( 'admin' ) . 'class-charitable-admin-blocks.php';

			require_once $this->get_path( 'admin' ) . 'class-charitable-admin.php';
			require_once $this->get_path( 'admin' ) . 'charitable-admin-hooks.php';

			// load campaign builder.
			require_once $this->get_path( 'admin' ) . 'campaign-builder/util.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/functions.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/ajax-actions.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/functions-legacy.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/debug.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/access.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/class-builder-fields.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/class-builder-form-fields.php';

			// connect for upgrade.wpcharitable.com.
			require_once $this->get_path( 'admin' ) . 'class-charitable-admin-connect.php';

			// misc.
			require_once $this->get_path( 'admin' ) . 'class-charitable-admin-pointers.php';
			require_once $this->get_path( 'admin' ) . 'plugins/charitable-admin-plugin-hooks.php';
			require_once $this->get_path( 'admin' ) . 'class-charitable-admin-getting-started.php';

			$admin = Charitable_Admin::get_instance();
			$this->registry->register_object( $admin );

			// Stripe.
			require_once $this->get_path( 'includes' ) . 'stripe/admin/class-charitable-stripe-admin.php';
			new Charitable_Stripe_Admin();

			return $admin;
		}

		/**
		 * Run the upgrade-server connect callback on init before any redirect (e.g. to login) can run.
		 * Only registered when CHARITABLE_CONNECT_CALLBACK is defined (request came from connect-callback.php).
		 *
		 * @since 1.8.9.6
		 */
		public function run_connect_callback() {
			ini_set( 'display_errors', '0' );
			require_once ABSPATH . 'wp-admin/includes/plugin.php';
			require_once ABSPATH . 'wp-admin/includes/file.php';
			require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
			$connect_file = $this->get_path( 'admin' ) . 'class-charitable-admin-connect.php';
			if ( file_exists( $connect_file ) ) {
				require_once $connect_file;
			}
			if ( ! class_exists( 'Charitable_Admin_Connect' ) ) {
				header( 'Content-Type: application/json' );
				wp_send_json( array( 'success' => false, 'data' => array( 'message' => 'Charitable not loaded.' ) ) );
			}
			Charitable_Admin_Connect::get_instance()->process();
			exit;
		}

		/**
		 * Checks whether we're on the public-facing side and if so, loads the public-facing functionality.
		 *
		 * @since  1.0.0
		 *
		 * @return Charitable_Public|false
		 */
		private function maybe_start_public() {
			if ( is_admin() && ! $this->is_ajax() ) {
				return false;
			}

			require_once $this->get_path( 'public' ) . 'class-charitable-public.php';

			// Campaign Builder related (v1.8.0).
			require_once $this->get_path( 'admin' ) . 'campaign-builder/class-campaign-builder-preview.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/class-builder-fields.php';
			require_once $this->get_path( 'admin' ) . 'campaign-builder/class-builder-form-fields.php';

			// load blocks/Gutenberg related.
			require_once $this->get_path( 'admin' ) . 'class-charitable-admin-blocks.php';

			$public = Charitable_Public::get_instance();

			$this->registry->register_object( $public );

			return $public;
		}

		/**
		 * Load the QUnit tests if ?qunit is appended to the request.
		 *
		 * @since  1.4.17
		 *
		 * @return boolean
		 */
		public function maybe_start_qunit() {
			/* Skip out early if ?qunit isn't included in the request. */
			if ( ! array_key_exists( 'qunit', $_GET ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended
				return false;
			}

		/* The unit tests have to exist. */
		if ( ! file_exists( $this->get_path( 'directory' ) . 'tests/qunit/tests.js' ) ) {
			return false;
		}

		/**
		 * Register QUnit testing library.
		 *
		 * Note: These files must be bundled locally for WordPress.org compliance.
		 * Download from:
		 * - JS: https://code.jquery.com/qunit/qunit-2.3.3.js
		 * - CSS: https://code.jquery.com/qunit/qunit-2.3.3.css
		 * Save to:
		 * - assets/js/libraries/qunit-2.3.3.js
		 * - assets/css/libraries/qunit-2.3.3.css
		 *
		 * @since 1.0.0
		 * @version 1.8.9.1
		 */
		wp_register_script(
			'qunit',
			$this->get_path( 'directory', false ) . 'assets/js/libraries/qunit-2.3.3.js',
			array(),
			'2.3.3',
			true
		);

		/* Version: '20170615-15:44' */
		wp_register_script( 'qunit-tests', $this->get_path( 'directory', false ) . 'tests/qunit/tests.js', array( 'jquery', 'qunit' ), time(), true );
		wp_enqueue_script( 'qunit-tests' );

		wp_register_style(
			'qunit',
			$this->get_path( 'directory', false ) . 'assets/css/libraries/qunit-2.3.3.css',
			array(),
			'2.3.3',
			'all'
		);
		wp_enqueue_style( 'qunit' );
		}

		/**
		 * Checks whether the current request is an AJAX request.
		 *
		 * @since  1.5.0
		 *
		 * @return boolean
		 */
		private function is_ajax() {
			return false !== ( defined( 'DOING_AJAX' ) && DOING_AJAX );
		}

		/**
		 * Checks whether we're executing an AJAX hook and if so, loads some AJAX functionality.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		private function maybe_start_ajax() {
			if ( ! $this->is_ajax() ) {
				return;
			}

			require_once $this->get_path( 'includes' ) . 'ajax/charitable-ajax-functions.php';
			require_once $this->get_path( 'includes' ) . 'ajax/charitable-ajax-hooks.php';
		}

		/**
		 * This method is fired after all plugins are loaded and simply fires the charitable_start hook.
		 *
		 * Extensions can use the charitable_start event to load their own functionality.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function charitable_start() {
			do_action( 'charitable_start', $this );
		}

		/**
		 * Fires off an action right after Charitable is installed, allowing other
		 * plugins/themes to do something at this point.
		 *
		 * @since   1.0.1
		 * @version 1.8.1.12 - Added transient for activation redirect.
		 * @version 1.8.3 - Added call to add version and delete notifications.
		 *
		 * @return void
		 */
		public function charitable_install() {
			$install = get_transient( 'charitable_install' );

			if ( ! $install ) {
				return;
			}

			require_once $this->get_path( 'includes' ) . 'plugin/class-charitable-install.php';

			Charitable_Install::finish_installing();

			do_action( 'charitable_install' );

			delete_transient( 'charitable_install' );

			// Store the date when the initial activation was performed.
			$type      = function_exists( 'charitable_is_pro' ) && charitable_is_pro() ? 'pro' : 'lite';
			$activated = (array) get_option( 'charitable_activated', array() );

			// If no activation date is set, assume it's a new install and set the activation redirect transient.
			if ( false === $activated || empty( $activated ) ) {
				// Add transient to trigger redirect to the Welcome / Getting Started screen.
				set_transient( 'charitable_activation_redirect', true, 30 );
			}

			$this->add_version_to_upgraded_from();

			if ( empty( $activated[ $type ] ) ) {
				$activated[ $type ] = time();
				update_option( 'charitable_activated', $activated );
			}

			// Clear notifications.
			delete_option( 'charitable_notifications' );
		}

		/**
		 * Returns whether we are currently in the start phase of the plugin.
		 *
		 * @since  1.0.0
		 *
		 * @return boolean
		 */
		public function is_start() {
			return current_filter() == 'charitable_start'; // phpcs:ignore
		}

		/**
		 * Returns whether the plugin has already started.
		 *
		 * @since  1.0.0
		 *
		 * @return boolean
		 */
		public function started() {
			return did_action( 'charitable_start' ) || current_filter() == 'charitable_start'; // phpcs:ignore
		}

		/**
		 * Returns whether the plugin is being activated.
		 *
		 * @since  1.0.0
		 *
		 * @return boolean
		 */
		public function is_activation() {
			return current_filter() == 'activate_charitable/charitable.php'; // phpcs:ignore
		}

		/**
		 * Returns whether the plugin is being deactivated.
		 *
		 * @since  1.0.0
		 *
		 * @return boolean
		 */
		public function is_deactivation() {
			return current_filter() == 'deactivate_charitable/charitable.php'; // phpcs:ignore
		}

		/**
		 * Returns plugin paths.
		 *
		 * @since  1.0.0
		 *
		 * @param  string  $type          If empty, returns the path to the plugin.
		 * @param  boolean $absolute_path If true, returns the file system path. If false, returns it as a URL.
		 * @return string
		 */
		public function get_path( $type = '', $absolute_path = true ) {
			$base = $absolute_path ? $this->directory_path : $this->directory_url;

			switch ( $type ) {
				case 'includes':
					$path = $base . 'includes/';
					break;

				case 'admin':
					$path = $base . 'includes/admin/';
					break;

				case 'public':
					$path = $base . 'includes/public/';
					break;

				case 'assets':
					$path = $base . 'assets/';
					break;

				case 'templates':
					$path = $base . 'templates/';
					break;

				case 'plugin-directory':
					$path = WP_PLUGIN_DIR;
					break;

				case 'directory':
					$path = $base;
					break;

				default:
					$path = __FILE__;

			}//end switch

			return $path;
		}

		/**
		 * Returns the plugin's version number.
		 *
		 * @since  1.0.0
		 *
		 * @return string
		 */
		public function get_version() {
			$version = self::VERSION;

			if ( false !== strpos( $version, '-' ) ) {
				$parts   = explode( '-', $version );
				$version = $parts[0];
			}

			return $version;
		}

		/**
		 * Adds previous versions to the upgraded_from option.
		 *
		 * @since  1.8.3
		 */
		public function add_version_to_upgraded_from() {

			$upgraded_from = get_option( 'charitable_version_upgraded_from', array() );

			if ( ! is_array( $upgraded_from ) ) {
				$upgraded_from = array();
			}

			$upgraded_from[] = $this->get_version();

			update_option( 'charitable_version_upgraded_from', $upgraded_from, false );
		}

		/**
		 * Get the campaign fields registry
		 *
		 * If the registry has not been set up yet, this will also
		 * perform the task of setting it up initially.
		 *
		 * This method is called on the plugins_loaded hook.
		 *
		 * @since  1.6.0
		 *
		 * @return Charitable_Campaign_Field_Registry
		 */
		public function campaign_fields() {
			if ( ! $this->registry->has( 'campaign_field_registry' ) ) {
				$campaign_fields = new Charitable_Campaign_Field_Registry();

				/* Register it immediately to avoid endless recursion. */
				$this->registry->register_object( $campaign_fields );

				$fields = include $this->get_path( 'includes' ) . 'fields/default-fields/campaign-fields.php';

				foreach ( $fields as $key => $args ) {
					$campaign_fields->register_field( new Charitable_Campaign_Field( $key, $args ) );
				}
			}

			return $this->registry->get( 'campaign_field_registry' );
		}

		/**
		 * Get the donation fields registry
		 *
		 * If the registry has not been set up yet, this will also
		 * perform the task of setting it up initially.
		 *
		 * This method is called on the plugins_loaded hook.
		 *
		 * @since  1.5.0
		 *
		 * @return Charitable_Donation_Field_Registry
		 */
		public function donation_fields() {
			if ( ! $this->registry->has( 'donation_field_registry' ) ) {
				$donation_fields = new Charitable_Donation_Field_Registry();

				/* Register it immediately to avoid endless recursion. */
				$this->registry->register_object( $donation_fields );

				$fields = include $this->get_path( 'includes' ) . 'fields/default-fields/donation-fields.php';

				foreach ( $fields as $key => $args ) {
					$donation_fields->register_field( new Charitable_Donation_Field( $key, $args ) );
				}
			}

			return $this->registry->get( 'donation_field_registry' );
		}

		/**
		 * Return the Endpoints API object.
		 *
		 * If the Endpoints API has not been set up yet, this will also
		 * perform the task of setting it up initially.
		 *
		 * This method is called on the plugins_loaded hook.
		 *
		 * @since  1.5.0
		 *
		 * @return Charitable_Endpoints
		 */
		public function endpoints() {
			if ( ! $this->registry->has( 'endpoints' ) ) {
				/**
				 * The order in which we register endpoints is important, because
				 * it determines the order in which the endpoints are checked to
				 * find whether they are the current page.
				 *
				 * Any endpoint that builds on another endpoint should be registered
				 * BEFORE the endpoint it builds on. In other words, move from
				 * most specific to least specific.
				 */
				$endpoints = new Charitable_Endpoints();
				$endpoints->register( new Charitable_Campaign_Donation_Endpoint() );
				$endpoints->register( new Charitable_Campaign_Widget_Endpoint() );
				$endpoints->register( new Charitable_Campaign_Endpoint() );
				$endpoints->register( new Charitable_Donation_Cancellation_Endpoint() );
				$endpoints->register( new Charitable_Donation_Processing_Endpoint() );
				$endpoints->register( new Charitable_Donation_Receipt_Endpoint() );
				$endpoints->register( new Charitable_Email_Preview_Endpoint() );
				$endpoints->register( new Charitable_Email_Verification_Endpoint() );
				$endpoints->register( new Charitable_Forgot_Password_Endpoint() );
				$endpoints->register( new Charitable_Reset_Password_Endpoint() );
				$endpoints->register( new Charitable_Registration_Endpoint() );
				$endpoints->register( new Charitable_Login_Endpoint() );
				$endpoints->register( new Charitable_Profile_Endpoint() );
				$endpoints->register( new Charitable_Webhook_Listener_Endpoint() );

				$this->registry->register_object( $endpoints );
			}

			return $this->registry->get( 'endpoints' );
		}

		/**
		 * Returns a registered object.
		 *
		 * @since  1.0.0
		 *
		 * @param  string $class The type of class to be retrieved.
		 * @return object
		 */
		public function get_registered_object( $class ) { // phpcs:ignore
			return $this->registry->get( $class );
		}

		/**
		 * Registers our donormeta table as a meta data table.
		 *
		 * @since  1.6.0
		 *
		 * @global WPDB $wpdb
		 * @return void
		 */
		public function register_donormeta_table() {
			global $wpdb;

			$wpdb->donormeta = $wpdb->prefix . 'charitable_donormeta';
		}

		/**
		 * Returns the model for one of Charitable's database tables.
		 *
		 * @since  1.0.0
		 *
		 * @param  string $table The database table to retrieve.
		 * @return Charitable_DB
		 */
		public function get_db_table( $table ) {
			$tables = $this->get_tables();

			if ( ! isset( $tables[ $table ] ) ) {
				charitable_get_deprecated()->doing_it_wrong(
					__METHOD__,
					sprintf( 'Invalid table %s passed', $table ),
					'1.0.0'
				);
				return null;
			}

			return $this->registry->get( $tables[ $table ] );
		}

		/**
		 * Return the filtered list of registered tables.
		 *
		 * @since  1.0.0
		 *
		 * @return string[]
		 */
		private function get_tables() {
			/**
			 * Filter the array of available Charitable table classes.
			 *
			 * @since 1.0.0
			 * @version 1.8.1
			 *
			 * @param array $tables List of tables as a key=>value array.
			 */
			return apply_filters(
				'charitable_db_tables',
				array(
					'campaign_donations'  => 'Charitable_Campaign_Donations_DB',
					'donors'              => 'Charitable_Donors_DB',
					'donormeta'           => 'Charitable_Donormeta_DB',
					'donation_activities' => 'Charitable_Donation_Activities_DB',
					'campaign_activities' => 'Charitable_Campaign_Activities_DB',
				)
			);
		}

		/**
		 * Maybe activate Charitable when a new site is added in a multisite network.
		 *
		 * @since  1.4.6
		 *
		 * @param  int $blog_id The blog to activate Charitable on.
		 * @return void
		 */
		public function maybe_activate_charitable_on_new_site( $blog_id ) {
			if ( is_plugin_active_for_network( basename( $this->directory_path ) . '/charitable.php' ) ) {
				switch_to_blog( $blog_id );
				$this->activate( false );
				restore_current_blog();
			}
		}

		/**
		 * Runs on plugin activation.
		 *
		 * @see    register_activation_hook
		 *
		 * @since  1.0.0
		 *
		 * @param  boolean $network_wide Whether to enable the plugin for all sites in the network
		 *                               or just the current site. Multisite only. Default is false.
		 * @return void
		 */
		public function activate( $network_wide = false ) {
			require_once $this->get_path( 'includes' ) . 'plugin/class-charitable-install.php';

			if ( is_multisite() && $network_wide ) {
				global $wpdb;

				foreach ( $wpdb->get_col( "SELECT blog_id FROM $wpdb->blogs" ) as $blog_id ) { // phpcs:ignore
					switch_to_blog( $blog_id );
					new Charitable_Install( $this->includes_path );
					restore_current_blog();
				}
			} else {
				new Charitable_Install( $this->includes_path );
			}

			// create or update the install date for future reference.
			update_option( 'wpcharitable_activated_datetime', current_time( 'timestamp' ) ); // phpcs:ignore WordPress.DateTime.CurrentTimeTimestamp.Requested

			// Set initial timestamps for rotating menu items if they're already installed
			$this->set_initial_rotating_menu_timestamps();
		}

		/**
		 * Runs on plugin deactivation.
		 *
		 * @see    register_deactivation_hook
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function deactivate() {
			require_once $this->get_path( 'includes' ) . 'plugin/class-charitable-uninstall.php';
			new Charitable_Uninstall();
		}

		/**
		 * If a charitable_action event is triggered, delegate the event using do_action.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function do_charitable_actions() {
			if ( isset( $_REQUEST['charitable_action'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended

				$action = esc_attr( $_REQUEST['charitable_action'] ); // phpcs:ignore

				/**
				 * Handle Charitable action.
				 *
				 * @since 1.0.0
				 */
				do_action( 'charitable_' . $action );
			}
		}

		/**
		 * Throw error on object clone.
		 *
		 * This class is specifically designed to be instantiated once. You can retrieve the instance using charitable()
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function __clone() {
			charitable_get_deprecated()->doing_it_wrong(
				__FUNCTION__,
				__( 'Cloning this object is forbidden.', 'charitable' ),
				'1.0.0'
			);
		}

		/**
		 * Disable unserializing of the class.
		 *
		 * @since  1.0.0
		 *
		 * @return void
		 */
		public function __wakeup() {
			charitable_get_deprecated()->doing_it_wrong(
				__FUNCTION__,
				__( 'Unserializing instances of this class is forbidden.', 'charitable' ),
				'1.0.0'
			);
		}

		/**
		 * DEPRECATED METHODS
		 */

		/**
		 * Load plugin compatibility files on plugins_loaded hook.
		 *
		 * @deprecated 1.8.0
		 *
		 * @since  1.4.18
		 * @since  1.5.0 Deprecated.
		 *
		 * @return void
		 */
		public function load_plugin_compat_files() {
			charitable_get_deprecated()->deprecated_function(
				__METHOD__,
				'1.5.0.',
				'charitable_load_compat_functions'
			);

			charitable_load_compat_functions();
		}

		/**
		 * Stores an object in the plugin's registry.
		 *
		 * @deprecated 1.8.0
		 *
		 * @since  1.0.0
		 * @since  1.5.0 Deprecated.
		 *
		 * @param  mixed $object Object to be registered.
		 * @return void
		 */
		public function register_object( $object ) { // phpcs:ignore
			charitable_get_deprecated()->deprecated_function(
				__METHOD__,
				'1.5.0',
				'charitable()->registry()->register_object()'
			);

			$this->registry->register_object( $object );
		}

		/**
		 * Returns the public class.
		 *
		 * @deprecated 1.8.0
		 *
		 * @since  1.0.0
		 * @since  1.5.0 Deprecated.
		 *
		 * @return Charitable_Public
		 */
		public function get_public() {
			charitable_get_deprecated()->deprecated_function(
				__METHOD__,
				'1.5.0',
				'charitable_get_helper'
			);

			return $this->registry->get( 'Charitable_Public' );
		}

		/**
		 * Returns the admin class.
		 *
		 * @deprecated 1.8.0
		 *
		 * @since  1.0.0
		 * @since  1.5.0 Deprecated.
		 *
		 * @return Charitable_Admin
		 */
		public function get_admin() {
			charitable_get_deprecated()->deprecated_function(
				__METHOD__,
				'1.5.0',
				'charitable_get_helper'
			);

			return $this->registry->get( 'Charitable_Admin' );
		}

		/**
		 * Return the current request object.
		 *
		 * @deprecated 1.8.0
		 *
		 * @since  1.0.0
		 * @since  1.5.0 Deprecated.
		 *
		 * @return Charitable_Request
		 */
		public function get_request() {
			charitable_get_deprecated()->deprecated_function(
				__METHOD__,
				'1.5.0',
				'charitable_get_helper'
			);

			return $this->registry->get( 'Charitable_Request' );
		}

		/**
		 * Returns the instance of Charitable_Currency.
		 *
		 * @deprecated 1.7.0
		 *
		 * @since 1.4.0 Deprecated.
		 */
		public function get_currency_helper() {
			charitable_get_deprecated()->deprecated_function( __METHOD__, '1.4.0', 'charitable_get_currency_helper' );
			return charitable_get_currency_helper();
		}

		/**
		 * Returns the instance of Charitable_Locations.
		 *
		 * @deprecated 1.6.0
		 *
		 * @since 1.2.0 Deprecated.
		 */
		public function get_location_helper() {
			charitable_get_deprecated()->deprecated_function( __METHOD__, '1.2.0', 'charitable_get_location_helper' );
			return charitable_get_location_helper();
		}

		/**
		 * Determine if we use Stripe core files or use an addon if that's installed and active. Checks for USE_NEW_STRIPE first.
		 *
		 * @since  1.7.0
		 *
		 * @return boolean
		 */
		public function load_core_stripe() {

			if ( false !== ( defined( 'USE_NEW_STRIPE' ) && USE_NEW_STRIPE ) ) {
				return true;
			}

			// check and see if the core Stripe AM gateway is an active gateway, and if it is not, don't load the core stripe JS, etc.
			if ( class_exists( 'Charitable_Gateways' ) && class_exists( 'Charitable_Gateway_Stripe_AM' ) && ! Charitable_Gateways::get_instance()->is_active_gateway( Charitable_Gateway_Stripe_AM::get_gateway_id() ) ) {
				return false;
			}

			return true;
		}

		/**
		 * Determine Stripe Connect is active.
		 *
		 * @since  1.7.0
		 *
		 * @return boolean
		 */
		public function is_stripe_connect_addon() {

			if ( in_array( 'charitable-stripe/charitable-stripe.php', apply_filters( 'active_plugins', get_option( 'active_plugins' ) ) ) ) { // phpcs:ignore
				return true;
			}

			return false;
		}

		/**
		 * Action links for the plugins page.
		 *
		 * @since  1.7.0
		 *
		 * @param  array   $actions    An array of existing actions.
		 * @param  string  $plugin_file The plugin file we are modifying.
		 * @param  boolean $action_links An array of action links to add.
		 * @return array              An array of action links.
		 */
		public function plugin_action_links( $actions, $plugin_file, $action_links = false ) {

			if ( ! charitable_is_pro() ) :

				// if this isn't pro, add the upgrade link which is visible on the plugins page until the plugin name.

				$action_links = array(
					'proupgrade' => array(
						// Translators: This is an action link users can click to purchase a license for All in One SEO Pro.
						'label' => __( 'Upgrade to Pro', 'charitable' ),
						'url'   => 'https://wpcharitable.com/lite-vs-pro/?utm_source=WordPress&utm_campaign=WP+Charitable&utm_medium=Action+Link+Plugins+Page&utm_content=Upgrade+to+Pro',
					),
				);

			endif;

			if ( isset( $actions['edit'] ) ) {
				unset( $actions['edit'] );
			}

			return $this->parse_action_links( $actions, $plugin_file, $action_links, 'before' );
		}

		/**
		 * Parse the action links.
		 *
		 * @since 1.7.0
		 *
		 * @param array  $actions   An array of existing actions.
		 * @param string $plugin_file The plugin file we are modifying.
		 * @param array  $action_links An array of action links to add.
		 * @param string $position  The position to add the action links.
		 * @return array
		 */
		protected function parse_action_links( $actions, $plugin_file, $action_links = array(), $position = 'after' ) {
			if ( empty( $this->plugin ) ) {
				$this->plugin = CHARITABLE_PLUGIN_BASENAME;
			}

			if ( $this->plugin === $plugin_file && ! empty( $action_links ) ) {
				foreach ( $action_links as $key => $value ) {
					$link = array(
						$key => '<a href="' . $value['url'] . '">' . $value['label'] . '</a>',
					);

					$actions = 'after' === $position ? array_merge( $actions, $link ) : array_merge( $link, $actions );
				}
			}

			return $actions;
		}

		/**
		 * Set up licensing for pro.
		 *
		 * @since  1.7.0
		 *
		 * @return void
		 */
		public function setup_licensing() {
			charitable_get_helper( 'licenses' )->register_licensed_product(
				self::NAME,
				self::AUTHOR,
				self::VERSION,
				__FILE__
			);
		}

		/**
		 * Deativates and/or warnings about any Charitable verison conflicts.
		 *
		 * @since  1.7.0
		 */
		public function check_for_version_conflicts() {

			// Check for Charitable Stripe < 1.5.0.
			if ( in_array( 'charitable-stripe/charitable-stripe.php', apply_filters( 'active_plugins', get_option( 'active_plugins' ) ) ) ) { // phpcs:ignore
				if ( class_exists( 'Charitable_Stripe' ) && version_compare( Charitable_Stripe::VERSION, '1.5.0', '<' ) ) { // phpcs:ignore
					// manually deactivate plugin.
					$found   = false;
					$current = get_option( 'active_plugins', array() );
					$key     = array_search( 'charitable-stripe/charitable-stripe.php', $current, true );
					if ( false !== $key ) {
						$found = true;

						unset( $current[ $key ] );
						update_option( 'active_plugins', $current );

						// translators: %s is the version number.
						$message = sprintf( __( '<p><strong style="color:red;">NOTICE:</strong> <strong>Charitable</strong> detected an out-of-date and incompatible version of <strong>Charitable Stripe</strong> addon. To avoid issues with Stripe donations, <strong>Charitable Stripe</strong> %s was deactivated. Please update <strong>Charitable Stripe</strong> to version 1.5.0 or higher.</p><p>You can get the latest version of <strong>Charitable Stripe</strong> through your <a href="https://www.wpcharitable.com/account/" target="_blank">Charitable account at wpcharitable.com</a>. All active licenses are accessible in the "My Downloads" tab.</p>', 'charitable' ), Charitable_Stripe::VERSION );

						wp_die( $message . ' <a href="' . admin_url( 'plugins.php' ) . '">Click here to return to the plugins screen.</a>' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
					}
					add_option( 'charitable_version_warning', $message );
				} else {
					delete_option( 'charitable_version_warning' );
				}
			}
		}

		/**
		 * Set initial timestamps for rotating menu items if they're already installed.
		 * This ensures the 7-day countdown starts immediately for already-installed plugins.
		 *
		 * @since 1.8.8
		 *
		 * @return void
		 */
		private function set_initial_rotating_menu_timestamps() {
			// Get the rotating menu items configuration
			$admin_pages = Charitable_Admin_Pages::get_instance();

			try {
				$reflection = new ReflectionClass( $admin_pages );
				$method = $reflection->getMethod( 'get_marketing_rotation_items' );
				$method->setAccessible( true );
				$items = $method->invoke( $admin_pages );
			} catch ( ReflectionException $e ) {
				// Fallback if reflection fails in future PHP versions
				$items = array();
			}

			$current_time = time();

			foreach ( $items as $item ) {
				$option_name = $item['option_name'] ?? '';
				$option_key = $item['option_key'] ?? '';
				$plugin_file = $item['plugin_file'] ?? '';

				if ( empty( $option_name ) || empty( $plugin_file ) ) {
					continue;
				}

				// Check if the plugin is installed (file exists in plugins directory)
				$plugin_path = WP_PLUGIN_DIR . '/' . $plugin_file;
				if ( file_exists( $plugin_path ) ) {
					// Plugin is installed, check if we already have a timestamp
					$existing_option = get_option( $option_name );
					$has_timestamp = false;

					if ( is_array( $existing_option ) ) {
						$has_timestamp = isset( $existing_option[ $option_key ] ) && is_numeric( $existing_option[ $option_key ] );
					} elseif ( is_numeric( $existing_option ) ) {
						$has_timestamp = true;
					}

					// Only set timestamp if one doesn't already exist
					if ( ! $has_timestamp ) {
						if ( is_array( $existing_option ) ) {
							$existing_option[ $option_key ] = $current_time;
							update_option( $option_name, $existing_option );
						} else {
							update_option( $option_name, array( $option_key => $current_time ) );
						}
					}
				}
			}
		}
	}

	$charitable = new Charitable();

endif;