Using Hooks and Filters

This guide provides detailed instructions on how to customize LearnPress functionality using hooks and filters without modifying the core plugin files.

Understanding Hooks and Filters

WordPress and LearnPress use a system of hooks and filters to allow developers to modify functionality and appearance:

  • Action Hooks – Allow you to add custom content or functionality at specific points in the code
  • Filter Hooks – Allow you to modify data or content before it’s displayed or processed

Action Hooks

Action hooks allow you to add custom content or functionality at specific points in the LearnPress templates.

How to Use Action Hooks

To use an action hook, add a function to your theme’s functions.php file and hook it to a specific action:

Basic Action Hook Usage

// Add to functions.php
function my_custom_function() {
    // Your code here
    echo 'This is my custom content';
}
add_action('learn-press/hook-name', 'my_custom_function');

Common Action Hooks

Hook Name Description Location
learn-press/before-main-content Executes before the main content area templates/content-course.php
learn-press/after-main-content Executes after the main content area templates/content-course.php
learn-press/before-course-content Executes before the course content templates/single-course/content.php
learn-press/after-course-content Executes after the course content templates/single-course/content.php
learn-press/before-checkout-form Executes before the checkout form templates/checkout/form.php
learn-press/after-checkout-form Executes after the checkout form templates/checkout/form.php
learn-press/before-course-item-content Executes before course item content templates/single-course/content-item.php
learn-press/after-course-item-content Executes after course item content templates/single-course/content-item.php
learn-press/before-profile-content Executes before profile content templates/profile/profile.php
learn-press/after-profile-content Executes after profile content templates/profile/profile.php

Practical Examples

Example 1: Adding Content Before Course Content

// Add to functions.php
function my_custom_before_course_content() {
    echo '<div class="my-custom-notice">';
    echo 'This course has been updated on ' . date('F j, Y') . '.';
    echo '</div>';
}
add_action('learn-press/before-course-content', 'my_custom_before_course_content');

Example 2: Adding a Custom Message After Checkout Form

// Add to functions.php
function my_custom_after_checkout_message() {
    echo '<div class="checkout-guarantee">';
    echo '<i class="fas fa-shield-alt"></i> ';
    echo 'We offer a 30-day money-back guarantee on all courses.';
    echo '</div>';
}
add_action('learn-press/after-checkout-form', 'my_custom_after_checkout_message');

Example 3: Adding Social Sharing Buttons After Course Content

// Add to functions.php
function add_social_sharing_buttons() {
    global $post;
    
    // Only show on course pages
    if (get_post_type() !== LP_COURSE_CPT) {
        return;
    }
    
    $share_url = urlencode(get_permalink());
    $share_title = urlencode(get_the_title());
    
    echo '<div class="course-social-sharing">';
    echo '<h4>Share this course:</h4>';
    echo '<a href="https://www.facebook.com/sharer/sharer.php?u=' . $share_url . '" target="_blank" class="facebook-share"><i class="fab fa-facebook-f"></i> Facebook</a>';
    echo '<a href="https://twitter.com/intent/tweet?url=' . $share_url . '&text=' . $share_title . '" target="_blank" class="twitter-share"><i class="fab fa-twitter"></i> Twitter</a>';
    echo '<a href="https://www.linkedin.com/shareArticle?mini=true&url=' . $share_url . '&title=' . $share_title . '" target="_blank" class="linkedin-share"><i class="fab fa-linkedin-in"></i> LinkedIn</a>';
    echo '</div>';
}
add_action('learn-press/after-course-content', 'add_social_sharing_buttons');

Filter Hooks

Filter hooks allow you to modify data or content before it’s displayed or processed.

How to Use Filter Hooks

To use a filter hook, add a function to your theme’s functions.php file and hook it to a specific filter:

Basic Filter Hook Usage

// Add to functions.php
function my_custom_filter_function($content) {
    // Modify $content here
    $content = 'Modified: ' . $content;
    return $content;
}
add_filter('learn-press/filter-name', 'my_custom_filter_function');

Common Filter Hooks

Hook Name Description Parameters
learn-press/course-price-html Modifies the HTML of the course price $price_html, $course
learn-press/course-content-summary Modifies the course content summary $content, $course
learn_press_get_template Modifies the template file path $template_name, $args, $template_path, $default_path
learn-press/course-tabs Modifies the course tabs $tabs, $course
learn-press/user-can-enroll-course Controls whether a user can enroll in a course $can_enroll, $course, $user
learn-press/profile-tabs Modifies the profile tabs $tabs, $user
learn-press/course-item-class Modifies the CSS classes for course items $classes, $item_id, $course_id
learn-press/filter-courses/type/fields Modifies the course type filter options $filter_types

Practical Examples

Example 1: Modifying Course Price Display

// Add to functions.php
function my_custom_course_price_html($price_html, $course) {
    // Add a "Limited Time Offer" text for courses with sale price
    if ($course->has_sale_price()) {
        $price_html = '<span class="limited-offer">Limited Time Offer!</span> ' . $price_html;
    }
    return $price_html;
}
add_filter('learn-press/course-price-html', 'my_custom_course_price_html', 10, 2);

Example 2: Adding Custom Course Tabs

// Add to functions.php
function add_custom_course_tab($tabs, $course) {
    // Add a new "Resources" tab
    $tabs['resources'] = array(
        'title'    => __('Resources', 'your-text-domain'),
        'priority' => 40,
        'callback' => 'display_course_resources_tab'
    );
    return $tabs;
}
add_filter('learn-press/course-tabs', 'add_custom_course_tab', 10, 2);

// Callback function to display tab content
function display_course_resources_tab() {
    // Get the current course
    $course = LP_Global::course();
    
    // Display content
    echo '<h3>Course Resources</h3>';
    echo '<ul>';
    echo '<li><a href="#">Resource 1</a></li>';
    echo '<li><a href="#">Resource 2</a></li>';
    echo '<li><a href="#">Resource 3</a></li>';
    echo '</ul>';
    
    // You can also get custom field data if you've added any
    // $resources = get_post_meta($course->get_id(), '_course_resources', true);
}

Example 3: Customizing Course Filters

// Add to functions.php
function add_custom_course_type($filter_types) {
    // Add a new course type "Blended"
    $filter_types['blended'] = __('Blended', 'your-text-domain');
    return $filter_types;
}
add_filter('learn-press/filter-courses/type/fields', 'add_custom_course_type');

Example 4: Modifying Profile Tabs

// Add to functions.php
function customize_profile_tabs($tabs, $user) {
    // Rename existing tab
    if (isset($tabs['courses'])) {
        $tabs['courses']['title'] = __('My Learning Path', 'your-text-domain');
    }
    
    // Add new tab
    $tabs['achievements'] = array(
        'title'    => __('Achievements', 'your-text-domain'),
        'priority' => 40,
        'callback' => 'display_user_achievements_tab'
    );
    
    // Remove a tab
    if (isset($tabs['settings'])) {
        unset($tabs['settings']);
    }
    
    return $tabs;
}
add_filter('learn-press/profile-tabs', 'customize_profile_tabs', 10, 2);

// Callback function for the new tab
function display_user_achievements_tab() {
    $user = LP_Profile::instance()->get_user();
    echo '<h3>Your Achievements</h3>';
    // Display achievements content
}

Advanced Customization with Hooks

Modifying Course Query

You can modify how courses are queried using the learn-press/query/course-args filter:

Example: Customizing Course Query

// Add to functions.php
function modify_course_query_args($args) {
    // Only show courses from a specific category
    $args['tax_query'] = array(
        array(
            'taxonomy' => 'course_category',
            'field'    => 'slug',
            'terms'    => 'featured-courses',
        ),
    );
    
    // Change the number of courses per page
    $args['posts_per_page'] = 12;
    
    return $args;
}
add_filter('learn-press/query/course-args', 'modify_course_query_args');

Customizing Enrollment Process

You can customize the enrollment process using various hooks:

Example: Custom Actions After Enrollment

// Add to functions.php
function custom_after_enrollment_actions($course_id, $user_id, $user_course) {
    // Send a custom email
    $user = get_user_by('id', $user_id);
    $course = learn_press_get_course($course_id);
    
    $subject = 'Welcome to ' . $course->get_title();
    $message = 'Hi ' . $user->display_name . ',\n\n';
    $message .= 'Thank you for enrolling in ' . $course->get_title() . '.\n';
    $message .= 'Here are some resources to help you get started:\n';
    $message .= '- Course overview: ' . get_permalink($course_id) . '\n';
    $message .= '- Our student community: https://example.com/community\n\n';
    $message .= 'Happy learning!\n';
    
    wp_mail($user->user_email, $subject, $message);
    
    // Add user to a specific group
    // Your custom code here
}
add_action('learn-press/user-enrolled-course', 'custom_after_enrollment_actions', 10, 3);

Customizing Quiz Behavior

You can customize quiz behavior using various hooks:

Example: Modifying Quiz Results Display

// Add to functions.php
function custom_quiz_result_message($message, $quiz, $user, $attempt) {
    $result = $attempt->get_results();
    $percent = $result['result'];
    
    if ($percent >= 90) {
        $message = 'Excellent work! You scored ' . $percent . '%. You have mastered this topic!';
    } elseif ($percent >= 70) {
        $message = 'Good job! You scored ' . $percent . '%. You have a solid understanding of this topic.';
    } elseif ($percent >= 50) {
        $message = 'You scored ' . $percent . '%. You have a basic understanding, but might need to review some concepts.';
    } else {
        $message = 'You scored ' . $percent . '%. We recommend reviewing the material and trying again.';
    }
    
    return $message;
}
add_filter('learn-press/quiz/result-message', 'custom_quiz_result_message', 10, 4);

Working with the New Features

Customizing Online/Offline Course Filters

With the new online/offline course filter feature, you can customize how courses are filtered:

Example: Modifying Course Type Labels

// Add to functions.php
function modify_course_type_labels($filter_types) {
    // Change the labels
    if (isset($filter_types['online'])) {
        $filter_types['online'] = __('Virtual Learning', 'your-text-domain');
    }
    
    if (isset($filter_types['offline'])) {
        $filter_types['offline'] = __('In-Person Classes', 'your-text-domain');
    }
    
    return $filter_types;
}
add_filter('learn-press/filter-courses/type/fields', 'modify_course_type_labels');

Customizing Profile Page Pagination

With the recent updates to pagination for quizzes on the profile page, you can customize the pagination settings:

Example: Changing Pagination Settings

// Add to functions.php
function modify_profile_quizzes_pagination($args) {
    // Change number of quizzes per page
    $args['per_page'] = 10;
    
    return $args;
}
add_filter('learn-press/profile/quizzes/query-args', 'modify_profile_quizzes_pagination');

Best Practices

  1. Use a child theme for your customizations to prevent losing changes during theme updates
  2. Prefix your function names to avoid conflicts with other plugins
  3. Check hook parameters to ensure you’re using the correct number and types of parameters
  4. Test thoroughly after LearnPress updates to ensure your customizations still work
  5. Use conditional logic to ensure your code only runs when needed

Warning: Always use hooks and filters instead of directly modifying plugin files. Direct modifications will be lost when the plugin is updated.

Conclusion

By using hooks and filters, you can extensively customize LearnPress functionality without modifying the core plugin files. This approach ensures that your customizations will not be lost when updating the plugin.

Need more help? Refer to the Overriding LearnPress Templates documentation for information about customizing the appearance of LearnPress through template overrides.