0; } /** * Check and save the first form creation time stamp. * If not already saved. * * @since 1.10.1 * @return void */ public static function save_first_form_creation_time_stamp() { if ( ! Helper::current_user_can() || self::is_first_form_created() || ! defined( 'SRFM_FORMS_POST_TYPE' ) || ! post_type_exists( SRFM_FORMS_POST_TYPE ) ) { return; } // Get the first form creation time from the database that is published. $query = new \WP_Query( [ 'post_type' => SRFM_FORMS_POST_TYPE, 'posts_per_page' => 1, 'orderby' => 'date', 'order' => 'ASC', 'fields' => 'ids', 'post_status' => 'publish', ] ); if ( ! empty( $query->posts ) && isset( $query->posts[0] ) ) { // Get the first post from the query result. $post_id = $query->posts[0]; // Get the post creation time in GMT. $creation_time = get_post_field( 'post_date_gmt', $post_id ); // Convert the creation time to a timestamp. $timestamp = strtotime( $creation_time ); if ( ! $timestamp ) { return; } Helper::update_srfm_option( 'first_form_created_at', $timestamp ); } } /** * Check if n days have passed since the first form creation. * This is used to determine if the dynamic nudges should be shown. * * @param int $days Number of days to check. * @since 1.10.1 * @return bool */ public static function check_first_form_creation_threshold( $days = 3 ) { $first_form_creation_time_stamp = self::get_first_form_creation_time_stamp(); if ( ! $first_form_creation_time_stamp ) { return false; // No forms created yet. } /** * Calculate the number of days since the first form was created. */ $days_from_creation = ( strtotime( current_time( 'mysql' ) ) - $first_form_creation_time_stamp ) / DAY_IN_SECONDS; // Return a boolean indicating if the number of days since creation is greater than the specified days. return $days_from_creation > $days; } /** * Show action on plugin page. * * @param array $links links. * @return array * @since 1.4.2 */ public function add_action_links( $links ) { if ( ! Helper::has_pro() ) { // Display upsell link if SureForms Pro is not installed. $upsell_link = add_query_arg( [ 'utm_medium' => 'plugin-list', ], Helper::get_sureforms_website_url( 'pricing' ) ); ob_start(); ?> post_type ) { return true; } return $use_block_editor; } /** * Enable Gutenberg for SureForms associated post types. * * @since 0.0.10 */ public function enable_gutenberg_for_sureforms() { /** * Check if the classic editor is enabled from Classic Editor plugin settings or Divi settings. */ if ( 'block' === get_option( 'classic-editor-replace' ) || 'on' === get_option( 'et_enable_classic_editor' ) ) { return; } $srfm_post_types = apply_filters( 'srfm_enable_gutenberg_post_types', [ SRFM_FORMS_POST_TYPE ] ); if ( in_array( get_current_screen()->post_type, $srfm_post_types, true ) ) { add_filter( 'use_block_editor_for_post_type', '__return_true', 110 ); add_filter( 'gutenberg_can_edit_post_type', '__return_true', 110 ); } } /** * Sureforms editor header styles. * * @since 0.0.1 */ public function enqueue_header_styles() { $current_screen = get_current_screen(); $file_prefix = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? '' : '.min'; $dir_name = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? 'unminified' : 'minified'; $css_uri = SRFM_URL . 'assets/css/' . $dir_name . '/'; /* RTL */ if ( is_rtl() ) { $file_prefix .= '-rtl'; } if ( 'sureforms_form' === $current_screen->id ) { wp_enqueue_style( SRFM_SLUG . '-editor-header-styles', $css_uri . 'header-styles' . $file_prefix . '.css', [], SRFM_VER ); } } /** * Add menu page. * * @return void * @since 0.0.1 */ public function add_menu_page() { $menu_slug = 'sureforms_menu'; $logo = file_get_contents( plugin_dir_path( SRFM_FILE ) . 'images/icon.svg' ); add_menu_page( __( 'SureForms', 'sureforms' ), __( 'SureForms', 'sureforms' ), self::$sureforms_page_default_capability, $menu_slug, static function () { }, 'data:image/svg+xml;base64,' . base64_encode( $logo ), // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.obfuscation_base64_encode 30 ); // Add the Dashboard Submenu. add_submenu_page( $menu_slug, __( 'Dashboard', 'sureforms' ), __( 'Dashboard', 'sureforms' ), self::$sureforms_page_default_capability, $menu_slug, [ $this, 'render_dashboard' ] ); } /** * Add Settings page. * * @return void * @since 0.0.1 */ public function settings_page() { $callback = [ $this, 'settings_page_callback' ]; add_submenu_page( 'sureforms_menu', __( 'Settings', 'sureforms' ), __( 'Settings', 'sureforms' ), self::$sureforms_page_default_capability, 'sureforms_form_settings', $callback ); // Get the current submenu page. $submenu_page = isset( $_GET['page'] ) ? sanitize_text_field( wp_unslash( $_GET['page'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- $_GET['page'] does not provide nonce. if ( ! isset( $_GET['tab'] ) && 'sureforms_form_settings' === $submenu_page ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- $_GET['page'] does not provide nonce. wp_safe_redirect( admin_url( 'admin.php?page=sureforms_form_settings&tab=general-settings' ) ); exit; } } /** * Open to Upgrade to Pro submenu link in new tab. * * @return void * @since 1.6.1 */ public function add_upgrade_to_pro_target_attr() { ?> 'submenu_link_upgrade', ], Helper::get_sureforms_website_url( 'upgrade' ) ); add_submenu_page( 'sureforms_menu', __( 'Upgrade', 'sureforms' ), __( 'Upgrade', 'sureforms' ), self::$sureforms_page_default_capability, $upgrade_url ); } /** * Add SMTP promotional submenu page. * * @return void * @since 1.7.1 */ public function add_suremail_page() { add_submenu_page( 'sureforms_menu', __( 'SMTP', 'sureforms' ), __( 'SMTP', 'sureforms' ), self::$sureforms_page_default_capability, 'sureforms_smtp', [ $this, 'suremail_page_callback' ] ); // Get the current submenu page. $submenu_page = isset( $_GET['page'] ) ? sanitize_text_field( wp_unslash( $_GET['page'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- $_GET['page'] does not provide nonce. // Check if SureMail is installed and active. if ( 'sureforms_smtp' === $submenu_page && file_exists( WP_PLUGIN_DIR . '/suremails/suremails.php' ) && is_plugin_active( 'suremails/suremails.php' ) ) { // Plugin is installed and active - redirect to SureMail dashboard. wp_safe_redirect( admin_url( 'options-general.php?page=suremail#/dashboard' ) ); exit; } } /** * SMTP promotional page callback. * * @return void * @since 1.7.1 */ public function suremail_page_callback() { ?>
'; } /** * Add notification badge to SureForms menu when there are new entries. * * @since 1.7.3 * @return void */ public function maybe_add_entries_badge() { if ( ! Helper::current_user_can() ) { return; } // If currently viewing the entries listing page, mark it as visited and skip the badge. if ( isset( $_GET['page'] ) && SRFM_ENTRIES === $_GET['page'] ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Only checking the page slug. $this->mark_entries_page_visit(); return; } $srfm_options = get_option( 'srfm_options', [] ); $last_visit = isset( $srfm_options['entries_last_visited'] ) ? absint( $srfm_options['entries_last_visited'] ) : 0; $new_entries = Entries::get_entries_count_after( $last_visit ); if ( $new_entries <= 0 ) { return; } global $menu; foreach ( $menu as $index => $item ) { if ( isset( $item[2] ) && 'sureforms_menu' === $item[2] ) { ob_start(); ?> $sub_item ) { if ( isset( $sub_item[2] ) && SRFM_ENTRIES === $sub_item[2] ) { ob_start(); ?> $sub_item ) { if ( isset( $sub_item[2] ) && SRFM_PAYMENTS === $sub_item[2] ) { ob_start(); ?> $settings_link_html, ] ); $links = array_merge( $plugin_links, $links ); } return $links; } /** * Sureforms block editor styles. * * @since 0.0.1 */ public function enqueue_styles() { $current_screen = get_current_screen(); global $wp_version; $file_prefix = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? '' : '.min'; $dir_name = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? 'unminified' : 'minified'; $css_uri = SRFM_URL . 'assets/css/' . $dir_name . '/'; $vendor_css_uri = SRFM_URL . 'assets/css/minified/deps/'; /* RTL */ if ( is_rtl() ) { $file_prefix .= '-rtl'; } // Enqueue editor styles for post and page. if ( SRFM_FORMS_POST_TYPE === $current_screen->post_type ) { wp_enqueue_style( SRFM_SLUG . '-editor', $css_uri . 'backend/editor' . $file_prefix . '.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-backend-blocks', $css_uri . 'blocks/default/backend' . $file_prefix . '.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-intl', $vendor_css_uri . 'intl/intlTelInput-backend.min.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-common', $css_uri . 'common' . $file_prefix . '.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-reactQuill', $vendor_css_uri . 'quill/quill.snow.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-single-form-modal', $css_uri . 'single-form-setting' . $file_prefix . '.css', [], SRFM_VER ); // if version is equal to or lower than 6.6.2 then add compatibility css. if ( version_compare( $wp_version, '6.6.2', '<=' ) ) { $srfm_inline_css = '.srfm-settings-modal .srfm-setting-modal-container .components-toggle-control .components-base-control__help{ margin-left: 4em; }'; wp_add_inline_style( SRFM_SLUG . '-single-form-modal', $srfm_inline_css ); } } wp_enqueue_style( SRFM_SLUG . '-form-selector', $css_uri . 'srfm-form-selector' . $file_prefix . '.css', [], SRFM_VER ); wp_enqueue_style( SRFM_SLUG . '-common-editor', SRFM_URL . 'assets/build/common-editor.css', [], SRFM_VER, 'all' ); } /** * Get Breadcrumbs for current page. * * @since 0.0.1 * @return array Breadcrumbs Array. */ public function get_breadcrumbs_for_current_page() { global $post, $pagenow; $breadcrumbs = []; if ( 'admin.php' === $pagenow && isset( $_GET['page'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- We don't need nonce verification here. $page_title = get_admin_page_title(); $breadcrumbs[] = [ 'title' => $page_title, 'link' => '', ]; } elseif ( $post && in_array( $pagenow, [ 'post.php', 'post-new.php', 'edit.php' ], true ) ) { $post_type_obj = get_post_type_object( get_post_type() ); if ( $post_type_obj ) { $post_type_plural = $post_type_obj->labels->name; $breadcrumbs[] = [ 'title' => $post_type_plural, 'link' => admin_url( 'edit.php?post_type=' . $post_type_obj->name ), ]; if ( 'edit.php' === $pagenow && ! isset( $_GET['page'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- We don't need nonce verification here. $breadcrumbs[ count( $breadcrumbs ) - 1 ]['link'] = ''; } else { $breadcrumbs[] = [ /* Translators: Post Title. */ 'title' => sprintf( __( 'Edit %1$s', 'sureforms' ), get_the_title() ), 'link' => get_edit_post_link( $post->ID ), ]; } } } else { $current_screen = get_current_screen(); if ( $current_screen && 'sureforms_form' === $current_screen->post_type ) { $breadcrumbs[] = [ 'title' => 'Forms', 'link' => '', ]; } else { $breadcrumbs[] = [ 'title' => '', 'link' => '', ]; } } return $breadcrumbs; } /** * Enqueue Admin Scripts. * * @return void * @since 0.0.1 */ public function enqueue_scripts() { $current_screen = get_current_screen(); global $wp_version; $file_prefix = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? '' : '.min'; $dir_name = defined( 'SRFM_DEBUG' ) && SRFM_DEBUG ? 'unminified' : 'minified'; $css_uri = SRFM_URL . 'assets/css/' . $dir_name . '/'; $is_rtl = is_rtl(); $rtl = $is_rtl ? '-rtl' : ''; /** * List of the handles in which we need to add translation compatibility. */ $script_translations_handlers = []; $onboarding_instance = Onboarding::get_instance(); $localization_data = [ 'site_url' => get_site_url(), 'breadcrumbs' => $this->get_breadcrumbs_for_current_page(), 'sureforms_dashboard_url' => admin_url( '/admin.php?page=sureforms_menu' ), 'plugin_version' => SRFM_VER, 'global_settings_nonce' => Helper::current_user_can() ? wp_create_nonce( 'wp_rest' ) : '', 'is_pro_active' => Helper::has_pro(), 'is_first_form_created' => self::is_first_form_created(), 'check_three_days_threshold' => self::check_first_form_creation_threshold(), 'check_eight_days_threshold' => self::check_first_form_creation_threshold( 8 ), 'pro_plugin_version' => Helper::has_pro() ? SRFM_PRO_VER : '', 'pro_plugin_name' => Helper::has_pro() && defined( 'SRFM_PRO_PRODUCT' ) ? SRFM_PRO_PRODUCT : 'SureForms Pro', 'sureforms_pricing_page' => Helper::get_sureforms_website_url( 'pricing' ), 'field_spacing_vars' => Helper::get_css_vars(), 'is_ver_lower_than_6_7' => version_compare( $wp_version, '6.6.2', '<=' ), 'integrations' => Helper::sureforms_get_integration(), 'rotating_plugin_banner' => Helper::get_rotating_plugin_banner(), 'ajax_url' => admin_url( 'admin-ajax.php' ), 'sf_plugin_manager_nonce' => wp_create_nonce( 'sf_plugin_manager_nonce' ), 'plugin_installer_nonce' => wp_create_nonce( 'updates' ), 'plugin_activating_text' => __( 'Activating...', 'sureforms' ), 'plugin_activated_text' => __( 'Activated', 'sureforms' ), 'plugin_activate_text' => __( 'Activate', 'sureforms' ), 'plugin_installing_text' => __( 'Installing...', 'sureforms' ), 'plugin_installed_text' => __( 'Installed', 'sureforms' ), 'is_rtl' => $is_rtl, 'onboarding_completed' => method_exists( $onboarding_instance, 'get_onboarding_status' ) ? $onboarding_instance->get_onboarding_status() : false, 'onboarding_redirect' => isset( $_GET['srfm-activation-redirect'] ), // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Nonce is not required for the activation redirection. 'pointer_nonce' => wp_create_nonce( 'sureforms_pointer_action' ), 'general_settings_url' => admin_url( '/options-general.php' ), 'payments' => apply_filters( 'srfm_admin_localize_payments_data', [ 'stripe_connected' => Stripe_Helper::is_stripe_connected(), 'stripe_mode' => Stripe_Helper::get_stripe_mode(), 'stripe_connect_url' => Stripe_Helper::get_stripe_settings_url(), 'currencies_data' => Payment_Helper::get_all_currencies_data(), 'zero_decimal_currencies' => Payment_Helper::get_zero_decimal_currencies(), 'webhook_url' => Stripe_Helper::get_webhook_url(), 'webhook_test_connected' => Stripe_Helper::is_webhook_configured( 'test', true ), 'webhook_live_connected' => Stripe_Helper::is_webhook_configured( 'live', true ), 'is_transaction_present' => Stripe_Helper::is_transaction_present(), 'payment_currency' => Payment_Helper::get_currency(), 'currency_sign_position' => Payment_Helper::get_currency_sign_position(), ] ), ]; $is_screen_sureforms_menu = Helper::validate_request_context( 'sureforms_menu', 'page' ); $is_screen_add_new_form = Helper::validate_request_context( 'add-new-form', 'page' ); $is_screen_sureforms_forms = Helper::validate_request_context( 'sureforms_forms', 'page' ); $is_screen_sureforms_form_settings = Helper::validate_request_context( 'sureforms_form_settings', 'page' ); $is_screen_sureforms_payments = Helper::validate_request_context( 'sureforms_payments', 'page' ); $is_screen_sureforms_entries = Helper::validate_request_context( SRFM_ENTRIES, 'page' ); $is_screen_sureforms_learn = Helper::validate_request_context( 'sureforms_learn', 'page' ); $is_post_type_sureforms_form = SRFM_FORMS_POST_TYPE === $current_screen->post_type; /** * Check if the current screen is the SureForms Menu and AI Auth Email is present then we will add user type as registered. * Compatibility with existing UI code that checks for this condition. */ if ( $is_screen_sureforms_menu ) { // If email is stored send the user type as registered else non-registered. $localization_data['srfm_ai_details'] = [ 'type' => ! empty( get_option( 'srfm_ai_auth_user_email' ) ) ? 'registered' : 'non-registered', ]; } if ( $is_screen_sureforms_menu || $is_post_type_sureforms_form || $is_screen_add_new_form || $is_screen_sureforms_forms || $is_screen_sureforms_form_settings || $is_screen_sureforms_entries || $is_screen_sureforms_payments || $is_screen_sureforms_learn ) { $asset_handle = '-dashboard'; wp_enqueue_style( SRFM_SLUG . $asset_handle . '-font', 'https://fonts.googleapis.com/css2?family=Inter:wght@400;500&display=swap', [], SRFM_VER ); $script_asset_path = SRFM_DIR . 'assets/build/dashboard.asset.php'; $script_info = file_exists( $script_asset_path ) ? include $script_asset_path : [ 'dependencies' => [], 'version' => SRFM_VER, ]; wp_enqueue_script( SRFM_SLUG . $asset_handle, SRFM_URL . 'assets/build/dashboard.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . $asset_handle, 'scIcons', [ 'path' => SRFM_URL . 'assets/build/icon-assets' ] ); $script_translations_handlers[] = SRFM_SLUG . $asset_handle; if ( class_exists( 'SRFM_PRO\Admin\Licensing' ) ) { $license_active = \SRFM_PRO\Admin\Licensing::is_license_active(); $localization_data['is_license_active'] = $license_active; // Updating current licensing status. $srfm_pro_license_status = get_option( 'srfm_pro_license_status', '' ); $current_license_status = $license_active ? 'licensed' : 'unlicensed'; if ( $current_license_status !== $srfm_pro_license_status ) { update_option( 'srfm_pro_license_status', $current_license_status ); } } $localization_data['security_settings_url'] = admin_url( '/admin.php?page=sureforms_form_settings&tab=security-settings&subpage=recaptcha' ); $localization_data['integration_settings_url'] = admin_url( '/admin.php?page=sureforms_form_settings&tab=integration-settings' ); wp_localize_script( SRFM_SLUG . $asset_handle, SRFM_SLUG . '_admin', apply_filters( SRFM_SLUG . '_admin_filter', $localization_data ) ); wp_enqueue_style( SRFM_SLUG . '-dashboard', SRFM_URL . 'assets/build/dashboard.css', [], SRFM_VER, 'all' ); } if ( $is_screen_sureforms_form_settings || $is_screen_sureforms_forms ) { wp_enqueue_style( SRFM_SLUG . '-settings', $css_uri . 'backend/settings' . $file_prefix . $rtl . '.css', [], SRFM_VER ); } // Enqueue styles for the entries page. if ( $is_screen_sureforms_entries ) { $asset_handle = '-entries'; wp_enqueue_script( SRFM_SLUG . $asset_handle, SRFM_URL . 'assets/build/entries.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . $asset_handle, SRFM_SLUG . '_admin', apply_filters( SRFM_SLUG . '_admin_filter', $localization_data ) ); $script_translations_handlers[] = SRFM_SLUG . $asset_handle; } // Enqueue scripts for the learn page. if ( $is_screen_sureforms_learn ) { $asset_handle = '-learn'; wp_enqueue_script( SRFM_SLUG . $asset_handle, SRFM_URL . 'assets/build/learn.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . $asset_handle, SRFM_SLUG . '_admin', apply_filters( SRFM_SLUG . '_admin_filter', $localization_data ) ); $script_translations_handlers[] = SRFM_SLUG . $asset_handle; } // Enqueue scripts for the forms page. if ( $is_screen_sureforms_forms ) { $asset_handle = '-forms'; $script_asset_path = SRFM_DIR . 'assets/build/forms.asset.php'; $script_info = file_exists( $script_asset_path ) ? include $script_asset_path : [ 'dependencies' => [], 'version' => SRFM_VER, ]; wp_enqueue_script( SRFM_SLUG . $asset_handle, SRFM_URL . 'assets/build/forms.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . $asset_handle, SRFM_SLUG . '_admin', apply_filters( SRFM_SLUG . '_admin_filter', $localization_data ) ); wp_enqueue_style( SRFM_SLUG . $asset_handle, SRFM_URL . 'assets/build/forms.css', [], SRFM_VER, 'all' ); $script_translations_handlers[] = SRFM_SLUG . $asset_handle; } // Enqueue scripts for the SureMail promotional page. $is_screen_sureforms_smtp = Helper::validate_request_context( 'sureforms_smtp', 'page' ); if ( $is_screen_sureforms_smtp ) { $asset_handle = 'suremail'; $script_asset_path = SRFM_DIR . 'assets/build/' . $asset_handle . '.asset.php'; $script_info = file_exists( $script_asset_path ) ? include $script_asset_path : [ 'dependencies' => [], 'version' => SRFM_VER, ]; wp_enqueue_script( SRFM_SLUG . '-suremail', SRFM_URL . 'assets/build/' . $asset_handle . '.js', $script_info['dependencies'], SRFM_VER, true ); wp_enqueue_style( SRFM_SLUG . '-suremail', SRFM_URL . 'assets/build/suremail.css', [], SRFM_VER, 'all' ); // Localize script for SureMail page. $suremail_localization_data = [ 'ajax_url' => admin_url( 'admin-ajax.php' ), 'admin_url' => admin_url(), 'suremail_url' => 'https://sureforms.com/suremail/', 'plugin_installer_nonce' => wp_create_nonce( 'updates' ), 'sfPluginManagerNonce' => wp_create_nonce( 'sf_plugin_manager_nonce' ), 'suremail_status' => file_exists( WP_PLUGIN_DIR . '/suremails/suremails.php' ) ? ( is_plugin_active( 'suremails/suremails.php' ) ? 'active' : 'installed' ) : 'not_installed', ]; wp_localize_script( SRFM_SLUG . '-suremail', SRFM_SLUG . '_admin', apply_filters( SRFM_SLUG . '_suremail_admin_filter', $suremail_localization_data ) ); $script_translations_handlers[] = SRFM_SLUG . '-suremail'; } // Admin Submenu Styles. wp_enqueue_style( SRFM_SLUG . '-admin', $css_uri . 'backend/admin' . $file_prefix . $rtl . '.css', [], SRFM_VER ); if ( $is_screen_sureforms_form_settings ) { $asset_handle = 'settings'; $script_asset_path = SRFM_DIR . 'assets/build/' . $asset_handle . '.asset.php'; $script_info = file_exists( $script_asset_path ) ? include $script_asset_path : [ 'dependencies' => [], 'version' => SRFM_VER, ]; wp_enqueue_script( SRFM_SLUG . '-settings', SRFM_URL . 'assets/build/' . $asset_handle . '.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . '-settings', SRFM_SLUG . '_admin', $localization_data ); $script_translations_handlers[] = SRFM_SLUG . '-settings'; } if ( $is_screen_add_new_form ) { wp_enqueue_style( SRFM_SLUG . '-template-picker', $css_uri . 'template-picker' . $file_prefix . $rtl . '.css', [], SRFM_VER ); $sureforms_admin = 'templatePicker'; $script_asset_path = SRFM_DIR . 'assets/build/' . $sureforms_admin . '.asset.php'; $script_info = file_exists( $script_asset_path ) ? include $script_asset_path : [ 'dependencies' => [], 'version' => SRFM_VER, ]; wp_enqueue_script( SRFM_SLUG . '-template-picker', SRFM_URL . 'assets/build/' . $sureforms_admin . '.js', $script_info['dependencies'], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . '-template-picker', SRFM_SLUG . '_admin', [ 'site_url' => get_site_url(), 'plugin_url' => SRFM_URL, 'admin_url' => admin_url( 'admin.php' ), 'new_template_picker_base_url' => admin_url( 'post-new.php?post_type=sureforms_form' ), 'capability' => Helper::current_user_can(), 'template_picker_nonce' => Helper::current_user_can() ? wp_create_nonce( 'wp_rest' ) : '', 'is_pro_active' => Helper::has_pro(), 'srfm_ai_usage_details' => AI_Helper::get_current_usage_details(), 'is_pro_license_active' => AI_Helper::is_pro_license_active(), 'srfm_ai_auth_user_email' => get_option( 'srfm_ai_auth_user_email' ), 'pricing_page_url' => Helper::get_sureforms_website_url( 'pricing' ), 'licensing_nonce' => wp_create_nonce( 'srfm_pro_licensing_nonce' ), ] ); $script_translations_handlers[] = SRFM_SLUG . '-template-picker'; } // Quick action sidebar. $default_allowed_quick_sidebar_blocks = apply_filters( 'srfm_quick_sidebar_allowed_blocks', [ 'srfm/input', 'srfm/email', 'srfm/textarea', 'srfm/checkbox', 'srfm/number', 'srfm/inline-button', 'srfm/advanced-heading', 'srfm/payment', ] ); if ( ! is_array( $default_allowed_quick_sidebar_blocks ) ) { $default_allowed_quick_sidebar_blocks = []; } $srfm_enable_quick_action_sidebar = get_option( 'srfm_enable_quick_action_sidebar' ); if ( ! $srfm_enable_quick_action_sidebar ) { $srfm_enable_quick_action_sidebar = 'disabled'; } $quick_sidebar_allowed_blocks = get_option( 'srfm_quick_sidebar_allowed_blocks' ); $quick_sidebar_allowed_blocks = ! empty( $quick_sidebar_allowed_blocks ) && is_array( $quick_sidebar_allowed_blocks ) ? $quick_sidebar_allowed_blocks : $default_allowed_quick_sidebar_blocks; $srfm_ajax_nonce = wp_create_nonce( 'srfm_ajax_nonce' ); if ( Helper::is_sureforms_admin_page() ) { wp_enqueue_script( SRFM_SLUG . '-quick-action-siderbar', SRFM_URL . 'assets/build/quickActionSidebar.js', [], SRFM_VER, true ); wp_localize_script( SRFM_SLUG . '-quick-action-siderbar', SRFM_SLUG . '_quick_sidebar_blocks', [ 'allowed_blocks' => $quick_sidebar_allowed_blocks, 'srfm_enable_quick_action_sidebar' => $srfm_enable_quick_action_sidebar, 'srfm_ajax_nonce' => $srfm_ajax_nonce, 'srfm_ajax_url' => admin_url( 'admin-ajax.php' ), ] ); $script_translations_handlers[] = SRFM_SLUG . '-quick-action-siderbar'; } /** * Enqueuing SureTriggers Integration script. * This script loads suretriggers iframe in Intergations tab. */ if ( $is_post_type_sureforms_form ) { wp_enqueue_script( SRFM_SLUG . '-suretriggers-integration', SRFM_SURETRIGGERS_INTEGRATION_BASE_URL . 'js/v2/embed.js', [], SRFM_VER, true ); } // Check $script_translations_handlers is not empty before calling the function. if ( ! empty( $script_translations_handlers ) ) { // Remove duplicates values from the array. $script_translations_handlers = array_unique( $script_translations_handlers ); foreach ( $script_translations_handlers as $script_handle ) { Helper::register_script_translations( $script_handle ); } } } /** * Form Template Picker Admin Body Classes * WordPress sometimes translates class names in the admin body tag, which can result in * incorrect or missing class names when rendering the admin pages. This function ensures * that essential class names are manually added to the body tag to maintain proper functionality. * * @since 0.0.1 * @param string $classes Space separated class string. */ public function admin_template_picker_body_class( $classes = '' ) { // Define an associative array of class names and their corresponding conditions. // Each condition checks whether a specific request context matches. $srfm_classes = [ 'sureforms_page_sureforms_entries' => Helper::validate_request_context( SRFM_ENTRIES, 'page' ), 'sureforms_page_sureforms_form_settings' => Helper::validate_request_context( 'sureforms_form_settings', 'page' ), 'srfm-template-picker' => Helper::validate_request_context( 'add-new-form', 'page' ), ]; $add_srfm_classes = ''; // Loop through the defined classes and conditions. foreach ( $srfm_classes as $class => $condition ) { // Check if the condition evaluates to true. if ( $condition ) { // Append the class to the existing classes string, followed by a space. $add_srfm_classes .= empty( $add_srfm_classes ) ? $class : ' ' . $class; } } // Append the new classes to the existing classes string. if ( ! empty( $add_srfm_classes ) ) { $classes .= ' ' . $add_srfm_classes; } // Return the updated list of classes. return $classes; } /** * Disable spectra's quick action bar in sureforms CPT. * * @param string $status current status of the quick action bar. * @since 0.0.2 * @return string */ public function restrict_spectra_quick_action_bar( $status ) { $screen = get_current_screen(); if ( 'disabled' !== $status && isset( $screen->id ) && 'sureforms_form' === $screen->id ) { $status = 'disabled'; } return $status; } /** * Register Pro compatibility notices early for React pages. * * This method runs on admin_init (priority 5) to ensure notices are * registered BEFORE admin_enqueue_scripts, so they're available when * wp_localize_script runs. * * Hooked - admin_init (priority 5) * * @return void * @since 2.5.0 */ public function register_pro_compatibility_notices() { // Early exit if Pro is not active, user lacks permissions, or Notice_Manager is unavailable. if ( ! Helper::has_pro() || ! Helper::current_user_can() || ! class_exists( 'SRFM\Admin\Notice_Manager' ) ) { return; } // Register version outdated notice for React pages. if ( ! version_compare( SRFM_PRO_VER, SRFM_PRO_RECOMMENDED_VER, '>=' ) ) { $pro_plugin_name = defined( 'SRFM_PRO_PRODUCT' ) ? SRFM_PRO_PRODUCT : 'SureForms Pro'; $react_outdated_message = sprintf( // translators: %1$s: SureForms version, %2$s: SureForms Pro Plugin Name, %3$s: SureForms Pro Version. esc_html__( 'SureForms %1$s requires minimum %2$s %3$s to work properly. Please update to the latest version.', 'sureforms' ), esc_html( SRFM_VER ), esc_html( $pro_plugin_name ), esc_html( SRFM_PRO_RECOMMENDED_VER ) ); \SRFM\Admin\Notice_Manager::register_notice( [ 'id' => 'sureforms-pro-version-outdated', 'variant' => 'warning', 'message' => $react_outdated_message, 'actions' => [ [ 'label' => esc_html__( 'Update Now', 'sureforms' ), 'url' => admin_url( 'update-core.php' ), 'variant' => 'primary', ], ], 'pages' => [ 'all' ], ] ); } } /** * Admin Notice Callback if sureforms pro is out of date. * * Hooked - admin_notices * * @return void * @since 1.0.4 */ public function srfm_pro_version_compatibility() { if ( ! Helper::has_pro() ) { return; } if ( empty( get_current_screen() ) ) { return; } if ( ! Helper::current_user_can() ) { return; } $srfm_pro_license_status = get_option( 'srfm_pro_license_status', '' ); /** * If the license status is not set then get the license status and update the option accordingly. * This will be executed only once. Subsequently, the option status is updated by the licensing class on license activation or deactivation. */ if ( empty( $srfm_pro_license_status ) && class_exists( 'SRFM_PRO\Admin\Licensing' ) ) { $srfm_pro_license_status = \SRFM_PRO\Admin\Licensing::is_license_active() ? 'licensed' : 'unlicensed'; update_option( 'srfm_pro_license_status', $srfm_pro_license_status ); } $pro_plugin_name = defined( 'SRFM_PRO_PRODUCT' ) ? SRFM_PRO_PRODUCT : 'SureForms Pro'; $message = ''; $url = admin_url( 'admin.php?page=sureforms_form_settings&tab=account-settings' ); if ( 'unlicensed' === $srfm_pro_license_status ) { ob_start(); ?>

', '', '' . esc_html( $pro_plugin_name ) . '' ); ?>

=' ) ) { ob_start(); ?>

', '' ); ?>

'srfm-plugin-review-notice', 'type' => '', 'message' => $this->build_notice_markup( esc_html__( 'Amazing! SureForms is powering your forms and submissions - let\'s keep growing together!', 'sureforms' ), esc_html__( 'If SureForms has been helpful, would you mind taking a moment to leave a 5-star review on WordPress.org?', 'sureforms' ), esc_url( 'https://wordpress.org/support/plugin/sureforms/reviews/?filter=5#new-post' ), esc_html__( 'Rate SureForms', 'sureforms' ), esc_html__( 'Maybe later', 'sureforms' ), esc_html__( 'I already did', 'sureforms' ), WEEK_IN_SECONDS, true ), 'repeat-notice-after' => WEEK_IN_SECONDS, 'show_if' => $this->maybe_display_rating_notice(), 'display-with-other-notices' => true, ] ); add_action( 'astra_notice_after_markup_srfm-plugin-review-notice', [ $this, 'enqueue_notice_response_script' ] ); } /** * Display a "Getting Started" admin notice for new users who haven't yet * reached the rating-notice milestone (3+ forms or 3+ entries). * * The Astra Notices library handles the 7-day delay via the * `display-notice-after` parameter. * * @since 2.5.2 * @return void */ public function display_srfm_getting_started_notice() { // Only show to admins. if ( ! Helper::current_user_can() ) { return; } // Allow the notice to be disabled programmatically. if ( ! apply_filters( 'srfm_show_getting_started_notice', true ) ) { return; } Astra_Notices::add_notice( [ 'id' => 'srfm-getting-started-notice', 'type' => '', 'message' => $this->build_notice_markup( esc_html__( 'SureForms is ready to power your forms — explore what\'s possible!', 'sureforms' ), esc_html__( 'Manage your forms, track submissions, and discover features like AI Form Builder, payment integrations, and more from the SureForms dashboard.', 'sureforms' ), esc_url( admin_url( 'admin.php?page=sureforms_menu' ) ), esc_html__( 'Go to Dashboard', 'sureforms' ), esc_html__( 'Maybe later', 'sureforms' ), esc_html__( 'I already know', 'sureforms' ), WEEK_IN_SECONDS ), 'repeat-notice-after' => WEEK_IN_SECONDS, 'show_if' => ! $this->maybe_display_rating_notice(), 'display-notice-after' => WEEK_IN_SECONDS, 'display-with-other-notices' => true, ] ); add_action( 'astra_notice_after_markup_srfm-getting-started-notice', [ $this, 'enqueue_notice_response_script' ] ); } /** * Enqueue the notice response analytics script. * * Called via the astra_notice_after_markup_{id} hook so the script * only loads when a SureForms notice is actually rendered. * * @since 2.5.2 * @return void */ public function enqueue_notice_response_script() { if ( wp_script_is( 'srfm-notice-response', 'enqueued' ) ) { return; } wp_enqueue_script( 'srfm-notice-response', SRFM_URL . 'admin/assets/js/notice-response.js', [], SRFM_VER, true ); wp_localize_script( 'srfm-notice-response', 'srfmNoticeResponse', [ 'ajaxurl' => admin_url( 'admin-ajax.php' ), 'nonce' => wp_create_nonce( 'srfm_notice_response' ), ] ); } /** * Handle the notice response AJAX request. * * Validates the request and records the analytics event * for the notice button that was clicked. * * @since 2.5.2 * @return void */ public function handle_notice_response() { if ( ! check_ajax_referer( 'srfm_notice_response', 'nonce', false ) ) { wp_send_json_error( [ 'message' => __( 'Invalid nonce.', 'sureforms' ) ], 403 ); } if ( ! Helper::current_user_can() ) { wp_send_json_error( [ 'message' => __( 'Unauthorized user.', 'sureforms' ) ], 403 ); } $notice_id = isset( $_POST['notice_id'] ) ? sanitize_text_field( wp_unslash( $_POST['notice_id'] ) ) : ''; $button = isset( $_POST['button'] ) ? sanitize_text_field( wp_unslash( $_POST['button'] ) ) : ''; $valid = [ 'srfm-getting-started-notice' => [ 'go_to_dashboard' => 'getting_started_notice_cta', 'maybe_later' => 'getting_started_notice_snooze', 'dismissed' => 'getting_started_notice_dismiss', ], 'srfm-plugin-review-notice' => [ 'rate_sureforms' => 'rating_notice_cta', 'maybe_later' => 'rating_notice_snooze', 'dismissed' => 'rating_notice_dismiss', ], ]; if ( ! isset( $valid[ $notice_id ][ $button ] ) ) { wp_send_json_error( [ 'message' => __( 'Invalid parameters.', 'sureforms' ) ], 400 ); } $event_name = $valid[ $notice_id ][ $button ]; Analytics_Events::track( $event_name, $button ); wp_send_json_success(); } /** * Disables the capabilities for WPForms to avoid conflicts when enqueueing * scripts and styles for WPForms. * * This function is intended to prevent any potential conflicts that may arise * when WPForms scripts and styles are enqueued. By disabling certain capabilities, * it ensures that WPForms does not interfere with other functionalities. * * @param bool $user_can A boolean indicating whether the user has the capability. * @return bool Returns true if the capabilities are successfully disabled, false otherwise. * @since 1.4.2 */ public function disable_wpforms_capabilities( $user_can ) { // Note: Nonce verification is intentionally omitted here as no database operations are performed. // The values of the $_REQUEST variables are strictly validated, ensuring security without the need for nonce verification. // phpcs:ignore WordPress.Security.NonceVerification.Recommended $post_id = ! empty( $_REQUEST['post'] ) && ! empty( $_REQUEST['action'] ) ? absint( $_REQUEST['post'] ) : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended $post_type = $post_id ? get_post_type( $post_id ) : sanitize_text_field( wp_unslash( $_REQUEST['post_type'] ?? '' ) ); return SRFM_FORMS_POST_TYPE === $post_type ? false : $user_can; } /** * Enqueueus the admin pointer script and styles. * * @return void * @since 1.8.0 */ public function enqueue_admin_pointer() { if ( ! $this->is_admin_pointer_visible() ) { return; } wp_enqueue_style( 'wp-pointer' ); wp_enqueue_script( 'wp-pointer' ); wp_enqueue_script( 'sureforms-admin-pointer', plugins_url( 'admin/assets/js/sureforms-pointer.js', SRFM_FILE ), [ 'wp-pointer', 'jquery' ], SRFM_VER, true ); wp_localize_script( 'sureforms-admin-pointer', 'sureformsPointerData', [ 'ajaxurl' => admin_url( 'admin-ajax.php' ), 'pointer_nonce' => wp_create_nonce( 'sureforms_pointer_action' ), ] ); } /** * Ajax handler for pointer popup visibility. * * @return void * @since 1.8.0 */ public function pointer_should_show() { // Security: Check user capability. if ( ! Helper::current_user_can() ) { wp_send_json_error( [ 'message' => __( 'Unauthorized user.', 'sureforms' ) ], 403 ); } // Security: Nonce check. if ( empty( $_POST['pointer_nonce'] ) || ! wp_verify_nonce( sanitize_text_field( wp_unslash( $_POST['pointer_nonce'] ) ), 'sureforms_pointer_action' ) ) { wp_send_json_error( [ 'message' => __( 'Invalid nonce.', 'sureforms' ) ], 403 ); } $content_markup = sprintf( /* translators: 1: opening span, 2: opening strong (inline), 3: closing strong, 4: closing span, 5: opening strong (block), 6: closing strong */ __( '%1$sGet started by %2$sbuilding your first form%3$s.%4$s%5$sExperience the power of our intuitive AI Form Builder%6$s', 'sureforms' ), '', '', '', '
', '', '' ); wp_send_json( [ 'show' => true, 'title' => esc_html( __( 'SureForms is waiting for you!', 'sureforms' ) ), 'content' => wp_kses_post( $content_markup ), 'button_text' => esc_html( __( 'Build My First Form', 'sureforms' ) ), 'dismiss' => esc_html( __( 'Dismiss', 'sureforms' ) ), 'button_url' => admin_url( 'admin.php?page=add-new-form' ), ] ); } /** * Ajax callback for pointer popup dismissed action. * * @return void * @since 1.8.0 */ public function pointer_dismissed() { // Security: Check user capability. if ( ! Helper::current_user_can() ) { wp_send_json_error( [ 'message' => __( 'Unauthorized user.', 'sureforms' ) ], 403 ); } // Security: Nonce check. if ( empty( $_POST['pointer_nonce'] ) || ! wp_verify_nonce( sanitize_text_field( wp_unslash( $_POST['pointer_nonce'] ) ), 'sureforms_pointer_action' ) ) { wp_send_json_error( [ 'message' => __( 'Invalid nonce.', 'sureforms' ) ], 403 ); } // Use Helper to update srfm_options key. Helper::update_srfm_option( 'pointer_popup_dismissed', time() ); wp_send_json_success(); } /** * Ajax pointer accepted CTA callback. * * @return void * @since 1.8.0 */ public function pointer_accepted_cta() { // Security: Check user capability. if ( ! Helper::current_user_can() ) { wp_send_json_error( [ 'message' => __( 'Unauthorized user.', 'sureforms' ) ], 403 ); } // Security: Nonce check. if ( empty( $_POST['pointer_nonce'] ) || ! wp_verify_nonce( sanitize_text_field( wp_unslash( $_POST['pointer_nonce'] ) ), 'sureforms_pointer_action' ) ) { wp_send_json_error( [ 'message' => __( 'Invalid nonce.', 'sureforms' ) ], 403 ); } // Use Helper to update srfm_options key. Helper::update_srfm_option( 'pointer_popup_accepted', time() ); wp_send_json_success(); } /** * Maybe register the dashboard widget based on entries. * * @return void * @since 1.9.1 */ public function maybe_register_dashboard_widget() { // Only for users with manage_options capability. if ( ! Helper::current_user_can() ) { return; } // Quick check if there are any entries in the last 7 days. $seven_days_ago = strtotime( '-7 days' ); $total_entries = Entries::get_entries_count_after( $seven_days_ago ); // Only add the dashboard setup hook if there are entries. if ( $total_entries > 0 ) { // Get forms with entries (limit 4 for dashboard widget). $this->dashboard_widget_data = Helper::get_forms_with_entry_counts( $seven_days_ago, 4 ); // Only show dashboard widget if there are forms with entries. if ( ! empty( $this->dashboard_widget_data ) ) { add_action( 'wp_dashboard_setup', [ $this, 'register_dashboard_widget' ] ); } } } /** * Register the dashboard widget. * * @return void * @since 1.9.1 */ public function register_dashboard_widget() { // Add the widget with high priority to position it at the top. wp_add_dashboard_widget( 'sureforms_recent_entries', __( 'SureForms', 'sureforms' ), [ $this, 'render_dashboard_widget' ], null, null, 'normal', 'high' ); } /** * Render the dashboard widget content. * * @return void * @since 1.9.1 */ public function render_dashboard_widget() { // Use the pre-fetched data to avoid duplicate queries. $entries_data = $this->dashboard_widget_data; // Display the widget content. ?>

render_dashboard_widget_footer( $entries_data ); ?>
%2$s
%3$s
%7$s %9$s %10$s
', $image_path, $heading, $message, $cta_url, esc_attr( $cta_class ), $cta_attrs, $cta_text, $snooze_duration, $snooze_text, $dismiss_text ); } /** * Callback for displaying the rating notice conditionally. * * Returns true if the user has 3 or more published forms or 3 or more form entries. * * @since 2.5.2 * @return bool */ private function maybe_display_rating_notice() { if ( null === $this->should_show_rating ) { $entries_count = Entries::get_total_entries_by_status( 'all' ); $form_count = wp_count_posts( SRFM_FORMS_POST_TYPE ); $this->should_show_rating = $entries_count >= self::RATING_NOTICE_THRESHOLD || Helper::get_integer_value( $form_count->publish ?? 0 ) >= self::RATING_NOTICE_THRESHOLD; } return $this->should_show_rating; } /** * Get random premium feature text. * * @return string Random feature text. * @since 1.9.1 */ private function get_random_premium_feature_text() { $features = [ __( 'Use Conditional Logic to show only what matters', 'sureforms' ), __( 'Split your form into steps to keep it easy', 'sureforms' ), __( 'Let people upload files directly to your form', 'sureforms' ), __( 'Turn responses into downloadable PDFs automatically', 'sureforms' ), __( 'Let users sign with a simple signature field', 'sureforms' ), __( 'Connect your form to other tools using webhooks', 'sureforms' ), __( 'Use Conversational Forms for a chat-like experience', 'sureforms' ), __( 'Let users register or log in through your form', 'sureforms' ), __( 'Build forms that create WordPress user accounts', 'sureforms' ), __( 'Add calculations to auto-total scores or prices', 'sureforms' ), ]; // Get a random feature. $random_key = array_rand( $features ); return $features[ $random_key ]; } /** * Render the dashboard widget footer for upsell. * * @param array $entries_data The entries data array. * @return void * @since 1.9.1 */ private function render_dashboard_widget_footer( $entries_data ) { // Only show footer if Pro is not active. if ( Helper::has_pro() ) { return; } // Count total entries in last 7 days. $total_entries = 0; foreach ( $entries_data as $form_data ) { $total_entries += $form_data['count']; } // Count total published forms. $published_forms_count = wp_count_posts( SRFM_FORMS_POST_TYPE )->publish; // Show footer only if 3+ entries received OR 3+ forms published. if ( $total_entries >= 3 || $published_forms_count >= 3 ) { ?> publish ?? 0 ) > 1 ) { return false; } if ( in_array( $pagenow, $allowed_pages, true ) ) { return true; } return false; } }