HEX
Server: LiteSpeed
System: Linux premium140.web-hosting.com 4.18.0-553.89.1.lve.el8.x86_64 #1 SMP Wed Dec 10 13:58:50 UTC 2025 x86_64
User: ukqcurpj (1011)
PHP: 8.1.34
Disabled: NONE
Upload Files
File: /home/ukqcurpj/www/wp-content/plugins/paid-memberships-pro/includes/compatibility/lifterlms.php
<?php
/**
 * Compatibility for the LifterLMS plugin.
 * @since 2.12
 * 
 * We add an option to "Streamline LifterLMS" to the
 * LifterLMS Setup Wizard and also the PMPro Advanced Settings Page.
 * When enabled, this will:
 * 1. Allow LifterLMS courses to be restricted by PMPro levels.
 * 2. Hide the LifterLMS Membership menu item from the admin dashboard.
 * 3. Hide the Restrictions tab of the edit course page.
 * 4. Hide the Access Plans section of the edit course page.
 * 5. Override the My Memberships and My Orders tabs of the student dashboard.
 */

/**
 * Add streamline setting to the PMPro Advanced Settings page.
 */
function pmpro_lifter_streamline_advanced_setting( $settings ) {
	$settings['lifter_streamline'] = [
		'field_name'  => 'lifter_streamline',
		'field_type'  => 'select',
		'options'	  => [
			'0' => __( 'No - All LifterLMS features are enabled.', 'paid-memberships-pro' ),
			'1' => __( 'Yes - Some LifterLMS features are disabled.', 'paid-memberships-pro' ),
		],
		'is_associative' => true,
		'label'       => __( 'Streamline LifterLMS', 'paid-memberships-pro' )		
	];
	return $settings;
}
add_filter( 'pmpro_custom_advanced_settings', 'pmpro_lifter_streamline_advanced_setting' );

/**
 * Add Require Membership box to LifterLMS courses.
 * @since 2.12.2 Removing lifterlms-membership-access metaboxes too.
 */
function pmpro_lifter_meta_boxes() {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}
	
	add_meta_box( 'pmpro_page_meta', esc_html__( 'Require Membership', 'paid-memberships-pro' ), 'pmpro_page_meta', 'course', 'side' );
	remove_meta_box( 'lifterlms-membership-access', array(), 'side' );
}
add_action( 'add_meta_boxes', 'pmpro_lifter_meta_boxes', 20 );

/**
 * Remove the LifterLMS Require Access Metaboxes.
 * @since 2.12.2
 */
function pmpro_lifter_restricted_post_types( $post_types ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $post_types;
	}
	
	// Note: This keeps LifterLMS from processing metaboxes, but actually results
	// in the metaboxes being added to EVERY CPT because the LLMS_Admin_Metabox::register
	// method doesn't expect the screens to be empty. So we remove all meta boxes above too.
	$post_types = array();
	return $post_types;
}
add_filter( 'llms_membership_restricted_post_types', 'pmpro_lifter_restricted_post_types' );

/**
 * Disable LifterLMS Membership restrictions by skipping checks
 * for posts and pages just in case a restriction was set
 * while Streamline was turned off.
 * Note: Other post types may still be restricted, but it's
 * unlikely if Streamline was always on.
 * @since 2.12.2
 */
function pmpro_lifter_membership_skip_post_types( $post_types ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $post_types;
	}
	
	$post_types[] = 'post';
	$post_types[] = 'page';
	$post_types = array_unique( $post_types );
	return $post_types;
}
add_filter( 'llms_is_post_restricted_by_membership_skip_post_types', 'pmpro_lifter_membership_skip_post_types' );

/**
 * Override PMPro's the_content filter.
 * We want to show course content even if it requires membership.
 * Still showing the non-member text at the bottom.
 */
function pmpro_lifter_membership_content_filter( $filtered_content, $original_content ) {	
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $filtered_content;
	}
	
	if ( is_singular( 'course' ) ) {
		// Show non-member text if needed.
		ob_start();
		// Get hasaccess ourselves so we get level ids and names.
		$hasaccess = pmpro_has_membership_access(NULL, NULL, true);
		if( is_array( $hasaccess ) ) {
			//returned an array to give us the membership level values
			$post_membership_levels_ids = $hasaccess[1];
			$post_membership_levels_names = $hasaccess[2];
			$hasaccess = $hasaccess[0];
			if ( ! $hasaccess ) {
				echo wp_kses_post( pmpro_get_no_access_message( '', $post_membership_levels_ids, $post_membership_levels_names ) );
			}
		}
		
		$after_the_content = ob_get_contents();
		ob_end_clean();			
		return $original_content . $after_the_content;		
	} else {
		return $filtered_content;	// Probably false.
	}
}
add_filter( 'pmpro_membership_content_filter', 'pmpro_lifter_membership_content_filter', 10, 2 );

/**
 * Get courses associated with a level.
 */
function pmpro_lifter_get_courses_for_levels( $level_ids ) {
	global $wpdb;
	
	// In case a level object was passed in.
	if ( is_object( $level_ids ) ) {
		$level_ids = $level_ids->ID;
	}
	
	// Make sure we have an array of ids.
	if ( ! is_array( $level_ids ) ) {
		$level_ids = array( $level_ids );
	}
	
	if ( empty( $level_ids ) ) {
		return array();
	}
	
	$level_placeholders = implode(',', array_fill(0, count($level_ids), '%d'));
	$course_ids = $wpdb->get_col(
		$wpdb->prepare(
			"
				SELECT mp.page_id 
				FROM $wpdb->pmpro_memberships_pages mp 
				LEFT JOIN $wpdb->posts p ON mp.page_id = p.ID 
				WHERE mp.membership_id IN($level_placeholders) 
				AND p.post_type = 'course' 
				AND p.post_status = 'publish' 
				GROUP BY mp.page_id
			",
			...$level_ids
		)
	);
	
	return $course_ids;
}

/**
 * Get all courses that are associated with any membership level.
 *
 * @since 3.6
 *
 * @return array<int> An array of course IDs.
 */
function pmpro_lifter_get_courses_for_all_levels() {
	static $course_ids = null;

	// If we don't have a cached value, query the database.
	if ( $course_ids == null ) {
		global $wpdb;
		$course_ids = $wpdb->get_col(
			"
				SELECT mp.page_id 
				FROM $wpdb->pmpro_memberships_pages mp 
				LEFT JOIN $wpdb->posts p ON mp.page_id = p.ID 
				WHERE p.post_type = 'course' 
				AND p.post_status = 'publish' 
				GROUP BY mp.page_id
			"
		);
	}	

	return $course_ids;
}

/**
 * Repair course enrollments for a user.
 *
 * @since 3.6
 *
 * @param int $user_id The user ID to repair enrollments for.
 */
function pmpro_lifter_repair_course_enrollments( $user_id ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}

	// Get all courses associated with any level.
	$all_level_courses = pmpro_lifter_get_courses_for_all_levels();

	// Get all courses for the user's current levels.
	$user_levels           = pmpro_getMembershipLevelsForUser( $user_id );
	$current_level_courses = pmpro_lifter_get_courses_for_levels( wp_list_pluck( $user_levels, 'ID' ) );

	// Get all the courses that the user is currently enrolled in.
	$student          = new LLMS_Student( $user_id );
	$enrolled_courses = ( $student->get_courses( array( 'status' => 'enrolled' ) ) )['results'];

	// Make sure all arrays are ints only to avoid type issues during comparisons.
	$all_level_courses     = array_map( 'intval', $all_level_courses );
	$current_level_courses = array_map( 'intval', $current_level_courses );
	$enrolled_courses      = array_map( 'intval', $enrolled_courses );

	// Unenroll from courses that are no longer associated with the user's levels.
	$courses_for_other_levels = array_diff( $all_level_courses, $current_level_courses );
	$courses_to_unenroll      = array_intersect( $enrolled_courses, $courses_for_other_levels );
	foreach ( $courses_to_unenroll as $course_id ) {
		// Not passing a $trigger here as this would only remove courses added by the 'pmpro' trigger.
		// In previous versions of PMPro, courses were not added with a trigger but we still want them to be removed going forwards.
		$student->unenroll( $course_id );
	}

	// Enroll in courses that are associated with the user's levels but not currently enrolled.
	$courses_to_enroll = array_diff( $current_level_courses, $enrolled_courses );
	foreach ( $courses_to_enroll as $course_id ) {
		// Using the 'pmpro' trigger to track which courses were added by PMPro.
		// In the future, this could be used so that we only unenroll from PMPro-added courses.
		$student->enroll( $course_id, 'pmpro' );
	}
}
add_action( 'pmpro_lifter_repair_course_enrollments', 'pmpro_lifter_repair_course_enrollments', 10, 1 );

/**
 * When users change levels, enroll/unenroll them from
 * any associated private courses.
 */
function pmpro_lifter_after_all_membership_level_changes( $pmpro_old_user_levels ) {		
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $pmpro_old_user_levels;
	}
	
	foreach ( $pmpro_old_user_levels as $user_id => $old_levels ) {
		pmpro_lifter_repair_course_enrollments( $user_id );
	}
}
add_action( 'pmpro_after_all_membership_level_changes', 'pmpro_lifter_after_all_membership_level_changes' );

/**
 * Callback to queue repair of each individual user's course enrollments.
 */
function pmpro_lifter_repair_all_course_enrollments_callback() {
	// Halt Action Scheduler processing to wait until we finish adding tasks.
	PMPro_Action_Scheduler::instance()->halt();

	$batch_size = 250;
	$offset = 0;
	do {
		$user_ids = get_users(
			array(
				'fields' => 'ID',
				'number' => $batch_size,
				'offset' => $offset,
			)
		);
		foreach ( $user_ids as $user_id ) {
			PMPro_Action_Scheduler::instance()->maybe_add_task(
				'pmpro_lifter_repair_course_enrollments',
				array( 'user_id' => $user_id ),
				'pmpro_async_tasks'
			);
		}
		$offset += $batch_size;
	} while ( count( $user_ids ) === $batch_size );

	// Resume Action Scheduler processing.
	PMPro_Action_Scheduler::instance()->resume();
}
add_action( 'pmpro_lifter_repair_all_course_enrollments_callback', 'pmpro_lifter_repair_all_course_enrollments_callback' );

/**
 * Queue Action Scheduler to repair course enrollments for all users.
 *
 * @since 3.6
 */
function pmpro_lifter_repair_all_course_enrollments() {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}

	// Halt Action Scheduler processing to wait until we finish adding tasks.
	PMPro_Action_Scheduler::instance()->maybe_add_task(
		'pmpro_lifter_repair_all_course_enrollments_callback',
		array(),
		'pmpro_async_tasks'
	);
}

/**
 * When the level restrictions for a post are updated, if the post was a course, repair the course enrollments for all users.
 *
 * @since 3.6
 *
 * @param int $post_id The ID of the post whose level restrictions were updated.
 */
function pmpro_lifter_after_updating_post_level_restrictions( $post_id ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}

	// Check if the post is a course.
	if ( get_post_type( $post_id ) !== 'course' ) {
		return;
	}

	// Queue the repair of course enrollments for all users.
	pmpro_lifter_repair_all_course_enrollments();
}
add_action( 'pmpro_after_updating_post_level_restrictions', 'pmpro_lifter_after_updating_post_level_restrictions' );

 /**
 * Hide the LifterLMS Membership menu item from the admin dashboard if streamline is enabled.
 */
function pmpro_lifter_hide_membership_menu() {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}
	
	// Remove the LifterLMS Membership menu item.
	remove_menu_page( 'edit.php?post_type=llms_membership' );
}
add_action( 'admin_menu', 'pmpro_lifter_hide_membership_menu', 99 );

/**
 * Hide the LifterLMS Membership and Checkout tabs from the admin settings.
 * @param array $tabs
 * @return array
 * @since 2.12
 */
function pmpro_lifter_hide_settings_tabs( $tabs ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $tabs;
	}

	if ( isset( $tabs['memberships'] ) ) {
		unset( $tabs['memberships'] );
	}

	if ( isset( $tabs['checkout'] ) ) {
		unset( $tabs['checkout'] );
	}

	return $tabs;
}
add_filter( 'lifterlms_settings_tabs_array', 'pmpro_lifter_hide_settings_tabs', 30 );

/**
 * Hide the LifterLMS memberships and sales tabs from the reporting page.
 */
function pmpro_lifter_reporting_tabs( $tabs ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $tabs;
	}

	if ( isset( $tabs['memberships'] ) ) {
		unset( $tabs['memberships'] );
	}

	if ( isset( $tabs['sales'] ) ) {
		unset( $tabs['sales'] );
	}
	
	return $tabs;
}
add_filter( 'lifterlms_reporting_tabs', 'pmpro_lifter_reporting_tabs' );

/**
 * Hide the Restrictions tab of the edit course page if streamline is enabled.
 */
function pmpro_lifter_hide_restrictions_tab( $fields ) {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $fields;
	}
	
	$new_fields = array();
	foreach( $fields as $field ) {
		if ( $field['title'] != __( 'Restrictions', 'lifterlms' ) ) {
			$new_fields[] = $field;
		}
	}
	
	return $new_fields;
}
add_filter( 'llms_metabox_fields_lifterlms_course_options', 'pmpro_lifter_hide_restrictions_tab' );

/**
 * Hide the "No Gateways" notice.
 * @since 2.12
 * @param bool $has_gateways
 * @return bool
 */
function pmpro_lifter_hide_no_payment_gateways_notice( $has_gateways ) {	
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $has_gateways;
	}

	// The usage here is odd, but returning true means there are gateways, so don't show the notice.
	return true;
}
add_filter( 'llms_admin_notice_no_payment_gateways', 'pmpro_lifter_hide_no_payment_gateways_notice' );

/**
 * Hide the Access Plans section of the edit course page if streamline is enabled.
 */
function pmpro_lifter_hide_access_plans() {
	// Bail if the streamline option is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}
	
	// Remove the Access Plans meta box.
	remove_meta_box( 'lifterlms-product', array( 'course', 'llms_membership' ), 'side' );
	remove_meta_box( 'lifterlms-product', array( 'course', 'llms_membership' ),  'normal' );
}
add_action( 'add_meta_boxes', 'pmpro_lifter_hide_access_plans', 20 );

/**
 * Override student dashboard links if streamline feature is enabled.
 */
function pmpro_lifter_override_dashboard_tabs( $tabs ) {	
	// Only override if streamlined is enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $tabs;
	}
	
	// Override the My Memberships tab.
	if ( isset( $tabs ) && isset( $tabs['view-memberships'] ) ) {
		$tabs['view-memberships']['url'] = pmpro_url( 'account' );
	}

	// Override the My Orders tab.
	if ( isset( $tabs ) && isset( $tabs['orders'] ) ) {
		$tabs['orders']['url'] = pmpro_url( 'invoice' );
	}

	return $tabs;
}
add_filter( 'llms_get_student_dashboard_tabs', 'pmpro_lifter_override_dashboard_tabs' );

/**
 * Insert our option into the intro step of the LifterLMS wizard.
 */
function pmpro_lifter_intro_html( $html, $wizard ) {
	// Get any previous streamline option. Default to true.
	$streamline = get_option( 'pmpro_lifter_streamline', null );	
	if ( $streamline === null ) {
		$streamline = 1;
	}
	
	// Save output buffer.
	ob_start();
	?>
	<hr />
	<p><?php esc_html_e( 'Since you already have Paid Memberships Pro installed, you can enable a "streamlined" version of LifterLMS that will let PMPro handle all checkouts, memberships, restrictions, and user fields.', 'paid-memberships-pro' ) ?></p>

	<label for="lifter-streamline">
		<input type="checkbox" name="lifter-streamline" id="lifter-streamline" <?php checked( (int)$streamline, 1 ); ?>>
		<?php esc_html_e( 'Enable streamlined version of LifterLMS', 'paid-memberships-pro' ) ?>
	</label>
	<script>
		jQuery(document).ready(function(){
			function pmpro_lifter_add_streamline_to_url( onload ) {
				let $checkbox = jQuery('#lifter-streamline');
				let $link = jQuery('.llms-setup-actions a.llms-button-primary');

				// If we are running this on load, add a nonce to the URL.
				if (onload) {
					$link.attr('href', $link.attr('href') + '&pmpro_lifter_streamline_nonce=<?php echo esc_attr( wp_create_nonce( 'pmpro_lifter_streamline_nonce' ) ); ?>');
				}

				//If the checkbox is checked, add streamline to the url
				$link.attr('href', $link.attr('href').replace('&pmpro_lifter_streamline=0', ''));
				$link.attr('href', $link.attr('href').replace('&pmpro_lifter_streamline=1', ''));
				if ($checkbox.is(':checked')) {
					$link.attr('href', $link.attr('href') + '&pmpro_lifter_streamline=1');
				} else {
					$link.attr('href', $link.attr('href') + '&pmpro_lifter_streamline=0');
				}
			}
			
			// Update the URL when the checkbox is changed.
			jQuery('#lifter-streamline').on('change', function(){
				pmpro_lifter_add_streamline_to_url( false );
			});

			// Run on load too.
			pmpro_lifter_add_streamline_to_url( true );
		});
	</script>
	<?php
	// Add the content buffer to the $html string.
	$html .= ob_get_clean();
	
	return $html;
}
add_filter( 'llms_setup_wizard_intro_html', 'pmpro_lifter_intro_html', 10, 2 );

/**
 * Hook into Page Setup step to save the streamline option.
 */
function pmpro_lifter_save_streamline_option() {
	// Bail if we're in the LifterLMS wizard.
	if ( empty( $_REQUEST['page'] ) || $_REQUEST['page'] !== 'llms-setup' ) {
		return;
	}

	// Bail if we're not on the pages step.
	if ( empty( $_REQUEST['step'] ) || $_REQUEST['step'] !== 'pages' ) {
		return;
	}

	// Bail if the current user doesn't have permission to manage LifterLMS.
	if ( ! current_user_can( 'manage_lifterlms' ) ) {
		return;
	}

	// Bail if a streamline option was not passed.
	if ( ! isset( $_REQUEST['pmpro_lifter_streamline'] ) ) {
		return;
	}

	// We are trying to set the streamline option. Check the nonce.
	if ( ! isset( $_REQUEST['pmpro_lifter_streamline_nonce'] ) || ! wp_verify_nonce( sanitize_key( $_REQUEST['pmpro_lifter_streamline_nonce'] ), 'pmpro_lifter_streamline_nonce' ) ) {
		wp_die( esc_html__( 'Security check failed.', 'paid-memberships-pro' ), '', array( 'response' => 403 ) );
	}

	// Get the streamline value.
	if ( ! empty( $_REQUEST['pmpro_lifter_streamline'] ) ) {
		$streamline = 1;
	} else {
		$streamline = 0;
	}
	
	update_option( 'pmpro_lifter_streamline', $streamline );
}
add_action( 'admin_init', 'pmpro_lifter_save_streamline_option' );

/**
 * If the streamline option is enabled, don't create some pages.
 */
function pmpro_lifter_install_get_pages( $pages ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $pages;
	}
	
	// Loop through and remove the membership catalog and checkout pages.
	$new_pages = array();
	foreach ( $pages as $page ) {
		if ( $page['slug'] == 'memberships' || $page['slug'] == 'purchase' ) {
			continue;
		}
		
		$new_pages[] = $page;
	}

	return $new_pages;
}
add_filter( 'llms_install_get_pages', 'pmpro_lifter_install_get_pages' );
add_filter( 'llms_install_create_pages', 'pmpro_lifter_install_get_pages' );	// Old filter.

/**
 * If the streamline option is enabled, unregister the LifterLMS membership post type.
 *
 * @param array $args The post type arguments.
 * @param string $post_type The post type name.
 * @since 3.1
 */
function pmpro_lifter_unregister_membership_post_type( $args, $post_type ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $args;
	}

	if ( 'llms_membership' === $post_type ) {
		$args['has_archive'] = false;
	}
	return $args;
}

add_action( 'register_post_type_args', 'pmpro_lifter_unregister_membership_post_type', 10, 2 );


/**
 * If (1) we are in streamlined mode and (2) PMPro Courses was
 * only being used for the LifterLMS module, deactivate it
 * and show a notice.
 */
function pmpro_lifter_maybe_deprecate_pmpro_courses( $deprecated_addons ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $deprecated_addons;
	}

	// Bail if PMPro Courses is not active.
	if ( ! defined( 'PMPRO_COURSES_VERSION') ) {
		return $deprecated_addons;
	}

	// Bail if any module besides the LifterLMS module is active.
	$active_modules = get_option( 'pmpro_courses_modules', array() );	
	if ( $active_modules !== array( 'lifterlms' ) ) {
		return $deprecated_addons;
	}

	// Okay, deprecate PMPro Courses
	$deprecated_addons['pmpro-courses'] = array(
		'file' => 'pmpro-courses.php',
		'label' => 'PMPro Courses',
		'message' => 'The Streamline LifterLMS option is enabled and includes all of the functionality previously found in the LifterLMS module of PMPro Courses.',
	);

	// Also deactivate the LifterLMS module, so folks can reactivate PMPro Courses if they want.
	update_option( 'pmpro_courses_modules', array() );	

	return $deprecated_addons;
}
add_filter( 'pmpro_deprecated_add_ons_list', 'pmpro_lifter_maybe_deprecate_pmpro_courses' );

/**
 * Filter the LifterLMS Admin Dashboard widgets.
 */
function pmpro_lifter_dashboard_widget_data( $data ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $data;
	}

	// Update the Net Sales widget.
	if ( isset( $data['sold'] ) ) {
		$data['sold']['link'] = admin_url( 'admin.php?page=pmpro-reports&report=sales' );
		$keys = array_keys( $data );
    	$keys[ array_search( 'sold', $keys ) ] = 'sold_pmpro';
    	$data = array_combine( $keys, $data );	
	}	

	return $data;
}
add_filter( 'llms_dashboard_widget_data', 'pmpro_lifter_dashboard_widget_data' );

/**
 * Filter some URLs on the LifterLMS dashboard.
 */
function pmpro_lifter_dashboard_admin_url( $url ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $url;
	}

	// Bail if not on the LifterLMS dashboard.
	if ( ! isset( $_REQUEST['page'] ) || $_REQUEST['page'] !== 'llms-dashboard' ) {
		return $url;
	}
	
	// Only update the following URLs.
	$swaps = array(
		'post-new.php?post_type=llms_membership' => 'admin.php?page=pmpro-membershiplevels',
		'edit.php?post_type=llms_order' => 'admin.php?page=pmpro-orders',
		'admin.php?page=llms-reporting&tab=sales' => 'admin.php?page=pmpro-reports&report=sales'
	);	
	foreach ( $swaps as $old => $new ) {
		if ( strpos( $url, $old ) !== false ) {
			$url = admin_url( $new );
			break;
		}
	}
	
	return $url;
}
add_filter( 'admin_url', 'pmpro_lifter_dashboard_admin_url' );

/**
 * Swap the Create Access Plan checklist item.
 */
function pmpro_lifter_dashboard_checklist( $checklist ) {
	global $wpdb;

	// Bail	if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $checklist;
	}

	// Bail if not on the LifterLMS dashboard.
	if ( ! isset( $_REQUEST['page'] ) || $_REQUEST['page'] !== 'llms-dashboard' ) {
		return $checklist;
	}

	// Swap the Create Access Plan checklist item.
	$sqlQuery = "SELECT COUNT(*) FROM $wpdb->pmpro_memberships_pages mp LEFT JOIN $wpdb->posts p ON mp.page_id = p.ID WHERE p.post_type = 'course' AND p.post_status = 'publish' GROUP BY mp.page_id LIMIT 1";
	$ap_check = $wpdb->get_var( $sqlQuery );
	if ( $ap_check ) {
		$checklist['access_plan'] = '<i class="fa fa-check"></i> ' . esc_html__( 'Restrict a Course', 'paid-memberships-pro' );
	} else {
		$checklist['access_plan'] = '<i class="fa fa-times"></i> <a href="https://www.paidmembershipspro.com/add-ons/lifterlms/?utm_source=plugin&utm_medium=lifterlms-dashboard&utm_campaign=lifterlms&utm_content=restrict-a-course" target="_blank" rel="noopener">' .  esc_html__( 'Restrict a Course', 'paid-memberships-pro' ) . '</a>';
	}	

	return $checklist;
}
add_filter( 'llms_dashboard_checklist', 'pmpro_lifter_dashboard_checklist' );

/**
 * AJAX callback to get PMPro sales data.
 */
function pmpro_lifter_ajax_llms_widget_sold_pmpro() {
	// Make sure the user has the pmpro_reports capability.
	if ( ! current_user_can( 'pmpro_reports' ) ) {
		wp_send_json_error( __( 'You do not have permission to perform this action.', 'paid-memberships-pro' ) );
	}

	// Get the start date.
	$startdate = sanitize_text_field( $_REQUEST['dates']['start'] );
	// Note: We ignore the enddate. pmpro_getRevenue assumes the enddate is today.

	// Get total PMPro sales in this period.
	$sales = llms_price_raw( floatval( pmpro_getRevenue( $startdate ) ) );

	// Format how LifterLMS expects it.
	$data = array(
		'chart_data' => array(
			'type'   => 'amount', // Type of field.			
			'header' => array(
				'id'    => 'sold_pmpro',
				'label' => __( 'Net Sales', 'paid-memberships-pro' ),
				'type'  => 'number',
			)
		),
		'charts'     => true,
		'message'    => 'success',
		'response'   => $sales,	// The response.
		'results'    => array(),
		'success'    => true
	);

	header( 'Content-Type: application/json' );
	echo wp_json_encode( $data );
	exit;
}
add_filter( 'wp_ajax_llms_widget_sold_pmpro', 'pmpro_lifter_ajax_llms_widget_sold_pmpro' );

/**
 * Make sure the PMPro lost password form
 * doesn't submit to the LifterLMS lost password form.
 *
 * @since 3.1.1
 */
function pmpro_maybe_remove_lifterlms_lostpassword_url_filter() {
	// Bail	if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return;
	}

	global $pmpro_pages;

	if ( ! empty( $pmpro_pages ) && ! empty( $pmpro_pages['login'] ) && is_page( $pmpro_pages['login'] ) ) {
		remove_filter( 'lostpassword_url', 'llms_lostpassword_url', 10, 0 );
	}
}
add_action( 'wp', 'pmpro_maybe_remove_lifterlms_lostpassword_url_filter' );

/**
 * Hide Membership Settings for an Access Plan for a course if LifterLMS streamline is enabled.
 */
function pmpro_llms_show_membership_settings_for_access_plans() {
	return ! get_option( 'pmpro_lifter_streamline' );
}
add_filter( 'llms_show_membership_settings_for_access_plans', 'pmpro_llms_show_membership_settings_for_access_plans' );

/**
 * Remove LifterLMS membership-related engagement triggers.
 */
function pmpro_lifterlms_engagement_triggers( $triggers ) {
	// Bail if streamline is not enabled.
	if ( ! get_option( 'pmpro_lifter_streamline' ) ) {
		return $triggers;
	}

	$new_triggers = [];

	$triggers_to_remove = array(
		'access_plan_purchased',
		'course_purchased',
		'membership_enrollment',
		'membership_purchased'
	);

    // Loop through the triggers and exclude membership-related ones.
    foreach ( $triggers as $key => $trigger ) {
        if ( ! in_array( $key, $triggers_to_remove ) ) {
            $new_triggers[ $key ] = $trigger;
        }
    }

    return $new_triggers;
}
add_filter( 'lifterlms_engagement_triggers', 'pmpro_lifterlms_engagement_triggers', 20 );