Terms & * Conditions" submenu is registered. Plugin action links are added for * both the standard and network admin plugins screens. * * @since 1.0.0 * @access public */ public function __construct() { if ( isset( self::$_this ) ) { wp_die( esc_html( sprintf( '%s is a singleton class and you cannot create a second instance.', get_class( $this ) ) ) ); } self::$_this = $this; // Enqueue admin styles and scripts on T&C plugin screens. add_action( 'admin_enqueue_scripts', array( $this, 'enqueue_assets' ) ); // Register the admin menu: nest under Complianz when available, otherwise use Tools. if ( ! defined( 'cmplz_version' ) ) { add_action( 'admin_menu', array( $this, 'register_main_menu' ), 20 ); } else { add_action( 'cmplz_admin_menu', array( $this, 'register_admin_page' ), 20 ); } // Add Settings and Support links to the Plugins list table. $plugin = cmplz_tc_plugin; add_filter( "plugin_action_links_$plugin", array( $this, 'plugin_settings_link' ) ); add_filter( "network_admin_plugin_action_links_$plugin", array( $this, 'plugin_settings_link' ) ); // Run the upgrade check and activation redirect on every admin_init. add_action( 'admin_init', array( $this, 'check_upgrade' ), 10, 2 ); add_action( 'admin_init', array( $this, 'maybe_redirect_to_settings' ), 10, 2 ); } /** * Returns the single instance of this class. * * @since 1.0.0 * @access public * * @return cmplz_tc_admin The singleton instance. */ public static function this() { return self::$_this; } /** * Redirects the user to the plugin settings page after first activation. * * Consumes the `cmplz_tc_redirect_to_settings` transient set by * `cmplz_tc_activation()`. The transient is deleted before the redirect * to ensure it fires only once and is not triggered by subsequent page * loads. Hooked to `admin_init`. * * @since 1.0.0 * @access public * * @see cmplz_tc_activation() Sets the transient on plugin activation. * * @return void */ public function maybe_redirect_to_settings() { if ( get_transient( 'cmplz_tc_redirect_to_settings' ) ) { // Delete the transient first so the redirect only happens once. delete_transient( 'cmplz_tc_redirect_to_settings' ); wp_safe_redirect( add_query_arg( array( 'page' => 'terms-conditions' ), admin_url( 'admin.php' ) ) ); exit; } } /** * Returns the install, upgrade, or "Installed" status link for a companion plugin. * * Determines the current state of a companion plugin by checking whether * its free or premium PHP constants are defined, then returns the * appropriate HTML anchor or plain-text label: * * - Neither free nor premium active → "Install" link to wordpress.org search. * - Free active but no premium → "Upgrade to pro" link to the plugin website. * - Premium (or wpsi_plugin) active → Plain "Installed" text (no action needed). * * @since 1.0.0 * @access public * * @see templates/wizard/other-plugins.php Calls this method for each plugin card. * * @param array $item Plugin descriptor array with keys: * - `constant_free` (string) PHP constant for the free edition. * - `constant_premium` (string) PHP constant for the premium edition. * - `search` (string) WordPress.org search query string. * - `website` (string) URL to the premium/pricing page. * @return string HTML anchor tag or plain-text status label. */ public function get_status_link( $item ) { $status = ''; if ( ! defined( $item['constant_free'] ) && ! defined( $item['constant_premium'] ) ) { // Plugin is not installed: link to the wordpress.org search results. $link = admin_url() . 'plugin-install.php?s=' . $item['search'] . '&tab=search&type=term'; $text = __( 'Install', 'complianz-terms-conditions' ); $status = '' . esc_html( $text ) . ''; } elseif ( 'wpsi_plugin' === $item['constant_free'] || defined( $item['constant_premium'] ) ) { // Premium (or a special-cased plugin) is active: no further action required. $status = esc_html__( 'Installed', 'complianz-terms-conditions' ); } elseif ( defined( $item['constant_free'] ) && ! defined( $item['constant_premium'] ) ) { // Free edition is active but premium is not: offer an upgrade link. $link = $item['website']; $text = __( 'Upgrade to pro', 'complianz-terms-conditions' ); $status = '' . esc_html( $text ) . ''; } return $status; } /** * Runs version-specific upgrade routines and updates the stored version number. * * Compares the previously stored plugin version against known migration * thresholds and applies any required data changes. After all migrations * are complete it fires the `cmplz_tc_upgrade` action (used by add-ons) * and writes the current version to the database. * * Note: when SCRIPT_DEBUG is enabled, a timestamp is appended to * `cmplz_tc_version`; the stored option value is compared without the * timestamp by reading it directly from the database. * * Hooked to: admin_init (priority 10). * * @since 1.0.0 * @access public * * @return void */ public function check_upgrade() { // Read the previously stored version; false when the plugin has never been upgraded. $prev_version = get_option( 'cmplz-tc-current-version', false ); if ( $prev_version && version_compare( $prev_version, '1.0.4', '<' ) ) { // Migration for < 1.0.4: re-save the documents update date to trigger a regeneration. update_option( 'cmplz_tc_documents_update_date', get_option( 'cmplz_tc_documents_update_date' ) ); } /** * Fires after version-specific upgrade routines have been applied. * * Add-ons can hook here to run their own migration logic. * * @since 1.0.0 * * @param string|false $prev_version The version string stored before this upgrade, or false on first run. */ do_action( 'cmplz_tc_upgrade', $prev_version ); // Persist the current version so future requests can detect upgrades. update_option( 'cmplz-tc-current-version', cmplz_tc_version ); } /** * Enqueues admin stylesheet and script on T&C plugin screens. * * Runs on `admin_enqueue_scripts` and bails immediately for any admin * page that is not part of the T&C plugin (detected by checking whether * `'terms-conditions'` appears in the hook suffix). In non-debug mode * the `.min` variants of all assets are loaded; in SCRIPT_DEBUG mode * the un-minified sources are used. The admin AJAX URL and a nonce are * passed to the script via `wp_localize_script()`. * * @since 1.0.0 * @access public * * @param string $hook The current admin page hook suffix * (e.g. `'tools_page_terms-conditions'`). * @return void */ public function enqueue_assets( $hook ) { // Only enqueue assets on screens that belong to this plugin. if ( false === strpos( $hook, 'terms-conditions' ) ) { return; } // Remove any conflicting Complianz GDPR wizard stylesheet. wp_dequeue_style( 'cmplz-wizard' ); // Use un-minified assets when SCRIPT_DEBUG is active. $minified = ( defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ) ? '' : '.min'; wp_register_style( 'cmplz-tc', trailingslashit( cmplz_tc_url ) . "assets/css/admin$minified.css", array(), cmplz_tc_version ); wp_enqueue_style( 'cmplz-tc' ); wp_register_style( 'cmplz-tc-tips-tricks', trailingslashit( cmplz_tc_url ) . "assets/css/tips-tricks$minified.css", array(), cmplz_tc_version ); wp_enqueue_style( 'cmplz-tc-tips-tricks' ); wp_enqueue_script( 'cmplz-tc-admin', cmplz_tc_url . "assets/js/admin$minified.js", array( 'jquery' ), cmplz_tc_version, true ); // Pass the AJAX endpoint URL and a save nonce to the admin JavaScript. wp_localize_script( 'cmplz-tc-admin', 'complianz_tc_admin', array( 'admin_url' => admin_url( 'admin-ajax.php' ), 'nonce' => wp_create_nonce( 'complianz_tc_save' ), ) ); } /** * Prepends Settings and Support links to the plugin's action links row. * * Hooked to both `plugin_action_links_{plugin}` (single-site) and * `network_admin_plugin_action_links_{plugin}` (multisite). The Support * link points to WordPress.org for the free edition and to * complianz.io/support for the premium edition. * * @since 1.0.0 * @access public * * @param array $links Existing action links for the plugin row. * @return array Modified links array with Settings and Support prepended. */ public function plugin_settings_link( $links ) { // Prepend the Settings link; placed after Support due to unshift order. $settings_link = '' . esc_html__( 'Settings', 'complianz-terms-conditions' ) . ''; array_unshift( $links, $settings_link ); // Link to WP.org support for free users; to complianz.io for premium. $support_link = defined( 'cmplz_free' ) ? 'https://wordpress.org/support/plugin/complianz-terms-conditions' : 'https://complianz.io/support'; $faq_link = '' . esc_html__( 'Support', 'complianz-terms-conditions' ) . ''; array_unshift( $links, $faq_link ); return $links; } /** * Registers a standalone "Terms & Conditions" submenu page under Tools. * * Used when the Complianz GDPR/CCPA plugin is not active. Capability- * gated via `cmplz_tc_user_can_manage()`. After adding the submenu the * `cmplz_admin_menu` action is fired so that add-ons can register their * own sub-items under this menu. * * Hooked to: admin_menu (priority 20). * * @since 1.0.0 * @access public * * @see cmplz_tc_user_can_manage() Capability check. * @see register_admin_page() Alternative used when Complianz GDPR is active. * * @return void */ public function register_main_menu() { if ( ! cmplz_tc_user_can_manage() ) { return; } global $cmplz_admin_page; $cmplz_admin_page = add_submenu_page( 'tools.php', // Parent menu slug. __( 'Terms & Conditions', 'complianz-terms-conditions' ), // Page title. __( 'Terms & Conditions', 'complianz-terms-conditions' ), // Menu label. 'manage_options', // Required capability. 'terms-conditions', // Menu slug. array( $this, 'wizard_page' ), // Page render callback. 40 // Menu position. ); /** * Fires after the standalone T&C admin menu has been registered. * * Add-ons hook here to append sub-items under the T&C menu entry. * * @since 1.0.0 */ do_action( 'cmplz_admin_menu' ); } /** * Registers a "Terms & Conditions" submenu page nested under the Complianz GDPR menu. * * Used when the Complianz GDPR/CCPA plugin is active and has already * registered its top-level `complianz` menu. Capability-gated via * `cmplz_tc_user_can_manage()`. * * Hooked to: cmplz_admin_menu (priority 20). * * @since 1.0.0 * @access public * * @see cmplz_tc_user_can_manage() Capability check. * @see register_main_menu() Alternative used when Complianz GDPR is absent. * * @return void */ public function register_admin_page() { if ( ! cmplz_tc_user_can_manage() ) { return; } add_submenu_page( 'complianz', // Parent menu slug (Complianz GDPR). __( 'Terms & Conditions', 'complianz-terms-conditions' ), // Page title. __( 'Terms & Conditions', 'complianz-terms-conditions' ), // Menu label. 'manage_options', // Required capability. 'terms-conditions', // Menu slug. array( $this, 'wizard_page' ) // Page render callback. ); } /** * Renders the wizard admin page for the Terms & Conditions document type. * * Delegates entirely to `cmplz_tc_wizard::wizard()`, which builds and * outputs the full wizard UI including the admin wrap, sidebar menu, * and step content. Registered as the `page_callback` for both * `register_main_menu()` and `register_admin_page()`. * * @since 1.0.0 * @access public * * @see cmplz_tc_wizard::wizard() Outputs the full wizard HTML. * * @return void */ public function wizard_page() { COMPLIANZ_TC::$wizard->wizard( 'terms-conditions' ); } /** * Outputs a tooltip icon with the provided help text as a data attribute. * * Renders a dashicon question-mark element whose tooltip content is set * via the `data-cmplz-tooltip` attribute and revealed by CSS/JS on hover. * The caller is responsible for escaping `$str` before passing it in. * * @since 1.0.0 * @access public * * @param string $str Pre-escaped help text to display in the tooltip. * @return void */ public function get_help_tip( $str ) { ?>