// phpcs:disable Generic.Commenting.DocComment.MissingShort
/** @noinspection PhpIllegalPsrClassPathInspection */
/** @noinspection AutoloadingIssuesInspection */
// phpcs:enable Generic.Commenting.DocComment.MissingShort
use WPForms\Admin\Notice;
if ( ! defined( 'ABSPATH' ) ) {
* Primary class constructor.
public function __construct() {
private function hooks(): void {
// Admin notice requesting review.
add_action( 'admin_init', [ $this, 'review_request' ] );
add_filter( 'admin_footer_text', [ $this, 'admin_footer' ], 1, 2 );
add_action( 'in_admin_footer', [ $this, 'promote_wpforms' ] );
* Add admin notices as needed for reviews.
public function review_request() {
// Only consider showing the review request to admin users.
// If the user has opted out of product announcement notifications, don't display the review request.
wpforms_setting( 'hide-announcements' ) ||
// Do not show the review request on Addons page.
wpforms_is_admin_page( 'addons' )
// Verify that we can do a check for reviews.
$notices = (array) get_option( 'wpforms_admin_notices', [] );
if ( empty( $notices['review_request'] ) ) {
$notices['review_request'] = [
update_option( 'wpforms_admin_notices', $notices );
// Check if it has been dismissed or not.
isset( $notices['review_request']['dismissed'], $notices['review_request']['time'] ) &&
! $notices['review_request']['dismissed'] &&
( ( $notices['review_request']['time'] + DAY_IN_SECONDS ) <= $time )
// If we cannot load, return early.
// The Logic is slightly different depending on what's at our disposal.
if ( class_exists( 'WPForms_Entry_Handler', false ) && wpforms()->is_pro() ) {
* Maybe show review request.
public function review() {
$entry_handler = wpforms()->obj( 'entry' );
$entries = $entry_handler ? $entry_handler->get_entries( [ 'number' => 50 ], true ) : 0;
// Only show review request if the site has collected at least 50 entries.
if ( empty( $entries ) || $entries < 50 ) {
// We have a candidate! Output a review message.
'dismiss' => Notice::DISMISS_GLOBAL,
'slug' => 'review_request',
'class' => 'wpforms-review-notice',
* Maybe show Lite review request.
public function review_lite() {
// Do not show the review request on Entries pages.
if ( wpforms_is_admin_page( 'entries' ) ) {
// Fetch when plugin was initially installed.
$activated = (array) get_option( 'wpforms_activated', [] );
if ( ! empty( $activated['lite'] ) ) {
// Only continue if the plugin has been installed for at least 14 days.
if ( ( $activated['lite'] + ( DAY_IN_SECONDS * 14 ) ) > time() ) {
$activated['lite'] = time();
update_option( 'wpforms_activated', $activated );
// Only proceed with displaying if the user created at least one form.
$form_count = wp_count_posts( 'wpforms' );
if ( empty( $form_count->publish ) ) {
// Check if the Constant Contact notice is displaying.
$cc = get_option( 'wpforms_constant_contact', false );
// If it's displaying don't ask for review until they configure CC or
// We have a candidate! Output a review message.
'dismiss' => Notice::DISMISS_GLOBAL,
'slug' => 'review_lite_request',
'class' => 'wpforms-review-notice',
* Output the review content.
private function review_content() {
<p><?php esc_html_e( 'Hey, there! It looks like you enjoy creating forms with WPForms. Would you do us a favor and take a few seconds to give us a 5-star review? We’d love to hear from you.', 'wpforms-lite' ); ?></p>
<a href="https://wordpress.org/support/plugin/wpforms-lite/reviews/?filter=5#new-post" class="wpforms-notice-dismiss wpforms-review-out" target="_blank" rel="noopener noreferrer"><?php esc_html_e( 'Ok, you deserve it', 'wpforms-lite' ); ?></a><br>
<a href="#" class="wpforms-notice-dismiss" target="_blank" rel="noopener noreferrer"><?php esc_html_e( 'Nope, maybe later', 'wpforms-lite' ); ?></a><br>
<a href="#" class="wpforms-notice-dismiss" target="_blank" rel="noopener noreferrer"><?php esc_html_e( 'I already did', 'wpforms-lite' ); ?></a>
* When a user is on a WPForms related admin page, display footer text
* that graciously asks them to rate us.
* @param string $text Footer text.
* @noinspection HtmlUnknownTarget
public function admin_footer( $text ) {
if ( ! empty( $current_screen->id ) && strpos( $current_screen->id, 'wpforms' ) !== false ) {
$url = 'https://wordpress.org/support/plugin/wpforms-lite/reviews/?filter=5#new-post';
wp_kses( /* translators: $1$s - WPForms plugin name, $2$s - WP.org review link, $3$s - WP.org review link. */
__( 'Please rate %1$s <a href="%2$s" target="_blank" rel="noopener noreferrer">★★★★★</a> on <a href="%3$s" target="_blank" rel="noopener">WordPress.org</a> to help us spread the word.', 'wpforms-lite' ),
'<strong>WPForms</strong>',
* Pre-footer promotion block, displayed on all WPForms admin pages except Form Builder.
public function promote_wpforms() {
// Some 3rd-party addons may use page slugs that start with `wpforms-` (e.g., WPForms Views),
// so we should define exact pages we want the footer to be displayed on instead
// of targeting any page that looks like a WPForms page.
// phpcs:ignore WordPress.Security.NonceVerification
$current_page = isset( $_REQUEST['page'] ) ? sanitize_text_field( wp_unslash( $_REQUEST['page'] ) ) : '';
if ( ! in_array( $current_page, $plugin_pages, true ) ) {
'url' => wpforms()->is_pro() ?
'https://wpforms.com/account/support/',
) : 'https://wordpress.org/support/plugin/wpforms-lite/',
'text' => __( 'Support', 'wpforms-lite' ),
'url' => wpforms_utm_link(
'https://wpforms.com/docs/',
'text' => __( 'Docs', 'wpforms-lite' ),
'url' => 'https://www.facebook.com/groups/wpformsvip/',
'text' => __( 'VIP Circle', 'wpforms-lite' ),
'url' => admin_url( 'admin.php?page=wpforms-about' ),
'text' => __( 'Free Plugins', 'wpforms-lite' ),
echo wpforms_render( // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
'title' => __( 'Made with ♥ by the WPForms Team', 'wpforms-lite' ),