WP admin custom forms – how to display errors? - Blog | createIT
Get a free advice now!

    Pick the topic
    Developer OutsourcingWeb developingApp developingDigital MarketingeCommerce systemseEntertainment systems

    Thank you for your message. It has been sent.
    Tags

    WP admin custom forms – how to display errors?

    WP admin custom forms – how to display errors?

    Do you use add_menu_page add_submenu_page in your plugin and have custom forms placed in the WordPress Dashboard Area? Having trouble in displaying form errors / success messages? In this article you will find a solution.

    Admin form custom $_GET param

    Let’s imagine we are authors of a WordPress plugin, and we would like to add a custom form in the Admin area. The form will provide an option to edit plugin settings. A typical solution will be:

    1. To create a custom admin page using add_menu_page or add_submenu_page
    2. To add a form with an action url being admin-post.php
    3. To use hidden input redirectToUrl to set a destination page (where to redirect after submit)
    4. On form submit – get $_POST data and save it in the database
    5. To display form success / or form errors above the form

    The form will be placed in the admin area, using menu_page url : /wp-admin/admin.php?page=ct-admin-forms . A popular solution for passing some data after submission is to append $_GET params to the url, example: /wp-admin/admin.php?page=ct-admin-forms?success=1 and use the success param to display the success message.

    However, this will not work in this situation. WordPress is validating all parameters on admin requests and will block access to it by displaying: Sorry, you are not allowed to access this page.

    Wp form submit – custom error message

    The solution will be to save the error message in the database using the set_transient() WordPress core function. The error message will be stored in the database for 30 seconds. After validation and redirecting to the form, we will use get_transient() to get and display the proper message.

    Now let’s put everything together and see a working example! Here is the registration of a new custom admin page:

    add_action('admin_menu',  'ct_add_menu_page', 20);
    function ct_add_menu_page()
    {
        add_menu_page(
            esc_html__('My menu section', 'ct-admin'),
            esc_html__('My menu section', 'ct-admin'),
            'manage_options',
            'ct-admin-forms',
            function () {
                include dirname(__FILE__) . '/inc/view0.php';
            },
            'dashicons-admin-page'
        );
    }

    Custom admin form template:

    <?php
    // /inc/view0.php
    ?>
    <h1><?php echo esc_html__('WP Admin custom form', 'ct-admin'); ?></h1>
    <?php include dirname(__FILE__) . '/inc/alerts.php'; ?>
    <form method="POST" action="<?php echo esc_html(admin_url('admin-post.php')); ?>">
        <input type="hidden" name="action" value="ct_admin_save">
        <?php wp_nonce_field('ct_admin_save', 'ct_admin'); ?>
        <input type="hidden" name="redirectToUrl" value="<?php echo admin_url('admin.php?page=ct-admin-forms') ?>">
        <?php
        /**
         * SOME FORM ELEMENTS HERE
         */
        ?>
        <div class="form__submit">
            <p class="submit">
                <input type="submit" name="submit5" id="submit5" class="button" value="Submit">
            </p>
        </div>
    </form>

    add_settings_error as transient

    The function for validating the form and saving $_POST data in the database. Before redirecting, we’re saving the error message as transient. ct_save_data_in_database() is just an imaginary function that is saving options to the wp_options table. In case of an error, it is returning an array of errors. Make sure to apply your logic there.

    add_action('admin_post_ct_admin_save','ct_submit_save');
    public function ct_submit_save()
    {
        /**
         * validate Nonce and user permissions
         */
        /**
         * save data to Database
         */
        $errors = ct_save_data_in_database();
        /**
         * save error message as transient and redirect
         */
        $redirect_to = $_POST['redirectToUrl'];
        if ($redirect_to) {
            if(empty($errors)){
                add_settings_error('ct_msg', 'ct_msg_option', __("Changes saved."), 'success');
            } else {
                add_settings_error('ct_msg', 'ct_msg_option', __("Unable to Submit Form: Please Try Again."), 'warning');
            }
            set_transient('settings_errors', get_settings_errors(), 30);
            wp_safe_redirect($redirect_to);
            exit;
        }
    }

    The last piece of the puzzle, a template for displaying errors. On page load, we’re deleting the transient that stores error messages.

    <?php
    /**
     * Display alerts after submitting custom admin wp form
     */
    // /inc/alerts.php
    function ct_admin_message($message, $msg_type = 'info') {
        return "<div id='message' class='alert alert-$msg_type'>$message</div>";
    }
    $form_errors = get_transient("settings_errors");
    delete_transient("settings_errors");
    if(!empty($form_errors)){
        foreach($form_errors as $error){
            echo ct_admin_message($error['message'], $error['type']);
        }
    }

    Types of alert messages

    We can add multiple errors and configure the type of message: success, info, warning or danger. For styling, we’re using the Bootstrap HTML markup (having bootstrap.css style enqueued, you will see alerts already styled).

    add_settings_error('ct_msg', 'ct_msg_option', __("Changes saved."), 'success');
    add_settings_error('ct_msg', 'ct_msg_option', __("Some info here"), 'info');
    add_settings_error('ct_msg', 'ct_msg_option', __("Error 1 here"), 'warning');
    add_settings_error('ct_msg', 'ct_msg_option', __("Error 2 here"), 'danger');

    That’s it for today’s tutorial. Be sure to follow us for other tips and guidelines, and don’t forget to subscribe to our newsletter.

    Comments
    1 response
    1. Hi Maciej!

      Thank you for this article! I noticed there is a typo in the URL of the second image above (the image from “Admin form custom $_GET param”). You got that error because you have “?success=1” instead of “&success=1”.

      Cheers,
      Vlad

    Add comment

    Your email address will not be published. Required fields are marked *

    Popular news

    Fetching Time records from ActiveCollab API
    • Dev Tips and Tricks

    Fetching Time records from ActiveCollab API

    September 9, 2024 by createIT
    Docker Compose for PrestaShop
    • Dev Tips and Tricks

    Docker Compose for PrestaShop

    September 2, 2024 by createIT
    WordPress wizard in admin – step by step
    • Dev Tips and Tricks

    WordPress wizard in admin – step by step

    August 29, 2024 by createIT
    Order Status Sync between PrestaShop and External APIs
    • Dev Tips and Tricks

    Order Status Sync between PrestaShop and External APIs

    August 26, 2024 by createIT
    What is PHP used for in web development 
    • Dev Tips and Tricks

    What is PHP used for in web development 

    August 22, 2024 by createIT
    Automating WooCommerce product availability date
    • Dev Tips and Tricks

    Automating WooCommerce product availability date

    August 15, 2024 by createIT
    WP Quiz Adventure – FAQ
    • Dev Tips and Tricks

    WP Quiz Adventure – FAQ

    August 12, 2024 by createIT
    Retrieval Augmented Generation tutorial and OpenAI example
    • Dev Tips and Tricks

    Retrieval Augmented Generation tutorial and OpenAI example

    August 8, 2024 by createIT
    10 useful SEO tools for the iGaming industry
    • Services
    • Technology

    10 useful SEO tools for the iGaming industry

    August 5, 2024 by createIT

    Technology
    Be on the same page as the rest of the industry.

    Contact us