Files
roi-theme/wp-content/plugins/wp-marketing-automations/modules/emails/class-bwfan-notification-email.php
root a22573bf0b Commit inicial - WordPress Análisis de Precios Unitarios
- WordPress core y plugins
- Tema Twenty Twenty-Four configurado
- Plugin allow-unfiltered-html.php simplificado
- .gitignore configurado para excluir wp-config.php y uploads

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-03 21:04:30 -06:00

600 lines
17 KiB
PHP
Executable File

<?php
#[AllowDynamicProperties]
class BWFAN_Notification_Email {
/**
* The single instance of the class.
*
* @var BWFAN_Notification_Email
*/
protected static $instance = null;
/**
* Global settings.
*
* @var array
*/
protected $global_settings = array();
/**
* Last executed notification.
*
* @var array
*/
protected $executed_last = array();
/**
* Constructor.
*/
public function __construct() {
$this->init();
}
/**
* Initialize the class.
*/
public function init() {
/** Schedule Email Notification WP cron event */
add_action( 'bwfan_after_save_global_settings', array( $this, 'set_scheduler' ), 10, 2 );
/** Email notification callback */
add_action( 'bwfan_run_notifications', array( $this, 'run_notifications' ) );
/** Testing */
add_action( 'admin_init', array( $this, 'test_notification_admin' ) );
add_action( 'wp_ajax_bwfan_send_test_email_notification', array( $this, 'send_test_email_notification' ) );
}
/**
* Get the instance of the class.
*
* @return BWFAN_Notification_Email
*/
public static function get_instance() {
if ( is_null( self::$instance ) ) {
self::$instance = new self();
}
return self::$instance;
}
/**
* Retrieves the HTML content of a template.
*
* This method includes the specified template file and allows passing arguments to it.
*
* @param $template
* @param $args
*
* @return false|string
*/
public static function get_template_html( $template, $args = array() ) {
if ( ! empty( $args ) && is_array( $args ) ) {
extract( $args ); // @codingStandardsIgnoreLine
}
ob_start();
include BWFAN_PLUGIN_DIR . '/templates/' . $template;
return ob_get_clean();
}
/**
* Set the scheduler for the BWFan notification email.
*
* This function is triggered when the 'bwfan_global_settings' option is updated.
* It checks if the 'bwfan_notification_time' value has changed and reschedules the 'bwfan_run_notifications' action accordingly.
*
* @param array $old_value The old value of the option.
* @param array $value The new value of the option.
*
* @return void
*/
public function set_scheduler( $old_value, $value ) {
if ( isset( $value['bwfan_enable_notification'] ) && $value['bwfan_enable_notification'] ) {
if ( isset( $old_value['bwfan_notification_time'] ) && $old_value['bwfan_notification_time'] !== $value['bwfan_notification_time'] ) {
if ( bwf_has_action_scheduled( 'bwfan_run_notifications' ) ) {
bwf_unschedule_actions( 'bwfan_run_notifications' );
}
}
if ( ! bwf_has_action_scheduled( 'bwfan_run_notifications' ) ) {
$notification_time = [];
if ( isset( $value['bwfan_notification_time'] ) && is_array( $value['bwfan_notification_time'] ) ) {
$notification_time = $value['bwfan_notification_time'];
}
$timestamp = $this->create_timestamp_from_array( $notification_time );
bwf_schedule_single_action( $timestamp, 'bwfan_run_notifications' );
}
return;
}
if ( bwf_has_action_scheduled( 'bwfan_run_notifications' ) ) {
bwf_unschedule_actions( 'bwfan_run_notifications' );
}
}
/**
* Create a timestamp from an array of time values.
*
* @param array $time_array An array of time values.
*
* @return int|bool The timestamp or false if required keys are missing.
*/
public function create_timestamp_from_array( $time_array ) {
// Check if required keys exist in the array
if ( isset( $time_array['hours'], $time_array['minutes'], $time_array['ampm'] ) ) {
$hours = intval( $time_array['hours'] );
$minutes = intval( $time_array['minutes'] );
$ampm = strtolower( $time_array['ampm'] );
if ( $ampm === 'am' && 12 === $hours ) {
$hours = 0;
} elseif ( $ampm === 'pm' && $hours < 12 ) {
// Convert 12-hour format to 24-hour format
$hours += 12;
}
return BWFAN_Common::get_store_time( $hours, $minutes, 0 );
}
return BWFAN_Common::get_store_time( 10 );
}
/**
* bwfan_run_notifications action callback
*
* @return void
* @throws DateMalformedStringException
*/
public function run_notifications() {
/** global settings */
$this->global_settings = BWFAN_Common::get_global_settings();
if ( false === $this->is_notification_active() ) {
return;
}
$frequencies = $this->get_frequencies();
if ( empty( $frequencies ) ) {
return;
}
/** Fetch the saved notifications data */
$this->executed_last = get_option( 'bwfan_email_notification_updated', array(
'daily' => '',
'weekly' => '',
'monthly' => '',
) );
$frequencies = $this->filter_frequencies( $frequencies );
if ( empty( $frequencies ) ) {
return;
}
$frequencies = self::prepare_frequencies( $frequencies );
if ( empty( $frequencies ) ) {
return;
}
foreach ( $frequencies as $frequency => $dates ) {
$this->send_email( $frequency, $dates );
}
}
/**
* Check if email notification is active.
*
* @return bool
*/
protected function is_notification_active() {
return isset( $this->global_settings['bwfan_enable_notification'] ) && $this->global_settings['bwfan_enable_notification'];
}
/**
* Get the frequencies for email notifications.
*
* @return array
*/
protected function get_frequencies() {
if ( isset( $this->global_settings['bwf_notification_frequency'] ) && is_array( $this->global_settings['bwf_notification_frequency'] ) ) {
return $this->global_settings['bwf_notification_frequency'];
}
return array();
}
/**
* Filter the frequencies based on the last saved option key.
*
* @param array $frequencies The frequencies to filter.
*
* @return array The filtered frequencies.
*/
protected function filter_frequencies( $frequencies = array() ) {
if ( empty( $frequencies ) ) {
return array();
}
/** Filter out the frequencies if an email was already sent */
return array_filter( $frequencies, function ( $frequency ) {
return ! $this->mail_sent( $frequency );
} );
}
/**
* Prepare frequencies
*
* @param $frequencies
*
* @return array
* @throws DateMalformedStringException
*/
public static function prepare_frequencies( $frequencies = [] ) {
$final = array();
if ( array_search( 'daily', $frequencies ) !== false ) {
$final['daily'] = BWFAN_Common::get_notification_day_range();
}
if ( array_search( 'weekly', $frequencies ) !== false ) {
$final['weekly'] = BWFAN_Common::get_notification_week_range();
}
if ( array_search( 'monthly', $frequencies ) !== false ) {
$final['monthly'] = BWFAN_Common::get_notification_month_range();
}
return $final;
}
/**
* Check if the email was sent for the given frequency.
*
* @param string $frequency The frequency to check.
*
* @return bool True if the email was sent, false otherwise.
*/
public function mail_sent( $frequency ) {
$today = new DateTime( 'now', wp_timezone() );
/** Check if the last execution time for the given frequency is not set */
if ( ! isset( $this->executed_last[ $frequency ] ) || empty( $this->executed_last[ $frequency ] ) ) {
return false;
}
try {
$last_sent = new DateTime( $this->executed_last[ $frequency ] );
$last_sent->setTimezone( wp_timezone() );
} catch ( Exception $e ) {
BWFAN_Common::log_test_data( "Frequency {$frequency} and value {$this->executed_last[ $frequency ]}", 'notification-error', true );
BWFAN_Common::log_test_data( "Exception {$e->getMessage()}", 'notification-error', true );
return false;
} catch ( Error $e ) {
BWFAN_Common::log_test_data( "Frequency {$frequency} and value {$this->executed_last[ $frequency ]}", 'notification-error', true );
BWFAN_Common::log_test_data( "Error {$e->getMessage()}", 'notification-error', true );
return false;
}
switch ( $frequency ) {
case 'daily':
return ! ( intval( $last_sent->format( 'Ymd' ) ) < intval( $today->format( 'Ymd' ) ) );
case 'weekly':
return ! ( intval( $last_sent->format( 'YW' ) ) < intval( $today->format( 'YW' ) ) );
case 'monthly':
return ! ( intval( $last_sent->format( 'Ym' ) ) < intval( $today->format( 'Ym' ) ) );
default:
return false;
}
}
/**
* Send email notification.
*
* @param $frequency
* @param $dates
*
* @return void
* @throws DateMalformedStringException
*/
public function send_email( $frequency, $dates ) {
/** Prepare metrics */
$metrics_controller = new BWFAN_Notification_Metrics_Controller( $dates, $frequency );
$metrics_controller->prepare_data();
/** Check if email has data */
if ( ! $metrics_controller->is_valid() ) {
return;
}
$data = $metrics_controller->get_data();
$email_controller = new BWFAN_Notification_Email_Controller( $frequency, $data, $dates );
/** Check if there are no recipients */
$to = $this->get_recipients();
if ( empty( $to ) ) {
return;
}
$subject = $this->get_email_subject( $frequency, $dates );
$body = $email_controller->get_content_html();
$headers = array( 'Content-Type: text/html; charset=UTF-8' );
$sent = false;
if ( is_array( $to ) && count( $to ) > 0 ) {
foreach ( $to as $recipient ) {
$sent = wp_mail( $recipient, $subject, $body, $headers );
}
}
/** Update the last execution time if the email was sent */
if ( $sent ) {
$this->executed_last[ $frequency ] = date( 'c' );
update_option( 'bwfan_email_notification_updated', $this->executed_last, false );
}
}
/**
* Get the recipients for the email.
*
* @return array The recipients for the email.
*/
private function get_recipients() {
$recipients = [];
if ( isset( $this->global_settings['bwf_notification_user_selector'] ) && is_array( $this->global_settings['bwf_notification_user_selector'] ) ) {
foreach ( $this->global_settings['bwf_notification_user_selector'] as $user ) {
if ( isset( $user['id'] ) && ! empty( $user['id'] ) ) {
$user_data = get_userdata( $user['id'] );
if ( $user_data ) {
$recipients[] = $user_data->user_email;
}
}
}
}
if ( isset( $this->global_settings['bwfan_external_user'] ) && is_array( $this->global_settings['bwfan_external_user'] ) ) {
foreach ( $this->global_settings['bwfan_external_user'] as $user ) {
if ( isset( $user['mail'] ) && ! empty( $user['mail'] ) ) {
$recipients[] = $user['mail'];
}
}
}
if ( empty( $recipients ) ) {
$recipients[] = get_option( 'admin_email' );
}
/** Filter array */
$recipients = array_filter( $recipients, function ( $email ) {
return ( strpos( $email, 'support@' ) === false );
} );
if ( is_array( $recipients ) ) {
$recipients = array_unique( $recipients );
sort( $recipients );
}
/**
* Filter the email recipients before returning
*
* @param array $recipients Array of email addresses
*/
return apply_filters( 'bwfan_notification_recipients', $recipients );
}
/**
* Get the email subject.
*
* @param $frequency
* @param $dates
*
* @return string
* @throws DateMalformedStringException
*/
public static function get_email_subject( $frequency, $dates ) {
$date_string = self::get_date_string( $dates, $frequency );
switch ( $frequency ) {
case 'daily':
/* translators: 1: Dynamic Text, 2: Dynamic Date */ return sprintf( __( '%1$s - Daily Report for %2$s', 'wp-marketing-automations' ), get_bloginfo( 'name' ), $date_string );
case 'weekly':
/* translators: 1: Dynamic Text, 2: Dynamic Date */ return sprintf( __( '%1$s - Weekly Report for %2$s', 'wp-marketing-automations' ), get_bloginfo( 'name' ), $date_string );
case 'monthly':
/* translators: 1: Dynamic Text, 2: Dynamic Date */ return sprintf( __( '%1$s - Monthly Report for %2$s', 'wp-marketing-automations' ), get_bloginfo( 'name' ), $date_string );
default:
return '';
}
}
/**
* Get the date string for the email subject.
*
* @param $dates
* @param $frequency
*
* @return string
* @throws DateMalformedStringException
*/
public static function get_date_string( $dates = array(), $frequency = 'weekly' ) {
if ( 'daily' === $frequency && isset( $dates['from_date'] ) ) {
return self::format_date( $dates['from_date'] );
}
if ( isset( $dates['from_date'] ) && isset( $dates['to_date'] ) ) {
/* translators: 1: Dynamic From Date, 2: Dynamic to date */
return sprintf( __( '%1$s - %2$s', 'wp-marketing-automations' ), self::format_date( $dates['from_date'] ), self::format_date( $dates['to_date'] ) );
}
return '';
}
/**
* Formats a date string to the desired format.
*
* @param $date_string
*
* @return string
* @throws DateMalformedStringException
*/
public static function format_date( $date_string ) {
/** Convert date string to a DateTime object */
$date = new DateTime( $date_string );
$date->setTimezone( wp_timezone() );
return $date->format( 'F j' );
}
/**
* Testing email notification.
*/
public function test_notification_admin() {
if ( false === current_user_can( 'administrator' ) ) {
return;
}
if ( ! isset( $_GET['bwfan_email_preview'] ) ) {
return;
}
$mode = filter_input( INPUT_GET, 'bwfan_mode', FILTER_SANITIZE_STRING );
$mode = empty( $mode ) ? 'weekly' : $mode;
switch ( $mode ) {
case 'monthly':
$range = BWFAN_Common::get_notification_month_range();
break;
case 'daily':
$range = BWFAN_Common::get_notification_day_range();
break;
default:
$range = BWFAN_Common::get_notification_week_range();
$mode = 'weekly';
break;
}
$dates = array(
'from_date' => $range['from_date'],
'to_date' => $range['to_date'],
'from_date_previous' => $range['from_date_previous'],
'to_date_previous' => $range['to_date_previous'],
);
/** Prepare metrics */
$metrics_controller = new BWFAN_Notification_Metrics_Controller( $dates, $mode );
$metrics_controller->prepare_data();
$data = $metrics_controller->get_data();
$email_controller = new BWFAN_Notification_Email_Controller( $mode, $data, $dates );
echo $email_controller->get_content_html(); //phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
exit;
}
/**
* Send test email notification.
*/
public function send_test_email_notification() {
if ( ! current_user_can( 'administrator' ) ) {
wp_send_json_error( __( 'You do not have permission to perform this action.', 'wp-marketing-automations' ) );
}
$mode = filter_input( INPUT_GET, 'bwfan_mode', FILTER_SANITIZE_STRING );
$mode = empty( $mode ) ? 'weekly' : $mode;
switch ( $mode ) {
case 'monthly':
$range = BWFAN_Common::get_notification_month_range();
break;
case 'daily':
$range = BWFAN_Common::get_notification_day_range();
break;
default:
$range = BWFAN_Common::get_notification_week_range();
$mode = 'weekly';
break;
}
$dates = array(
'from_date' => $range['from_date'],
'to_date' => $range['to_date'],
'from_date_previous' => $range['from_date_previous'],
'to_date_previous' => $range['to_date_previous'],
);
/** Prepare metrics */
$metrics_controller = new BWFAN_Notification_Metrics_Controller( $dates, $mode );
$metrics_controller->prepare_data();
$data = $metrics_controller->get_data();
$email_controller = new BWFAN_Notification_Email_Controller( $mode, $data, $dates );
$to = get_option( 'admin_email' );
$subject = self::get_email_subject( $mode, $dates );
$body = $email_controller->get_content_html();
$headers = array( 'Content-Type: text/html; charset=UTF-8' );
$sent = wp_mail( $to, $subject, $body, $headers );
if ( $sent ) {
wp_send_json_success( __( 'Email sent successfully.', 'wp-marketing-automations' ) );
} else {
wp_send_json_error( __( 'Failed to send email.', 'wp-marketing-automations' ) );
}
}
/**
* Function to set the default settings for notification settings
*
* @return void
*/
public static function set_bwfan_settings() {
$bwfan_settings = BWFAN_Common::get_global_settings();
if ( isset( $bwfan_settings['bwfan_enable_notification'] ) && ! empty( $bwfan_settings['bwfan_enable_notification'] ) ) {
return;
}
$new_settings = array(
'bwfan_enable_notification' => true,
'bwf_notification_frequency' => array( 'weekly', 'monthly' ),
'bwfan_notification_time' => array(
'hours' => '10',
'minutes' => '00',
'ampm' => 'am',
),
'bwfan_external_user' => array(),
);
$bwfan_settings = array_merge( $bwfan_settings, $new_settings );
$users = get_users( array( 'role' => 'administrator' ) );
$user_selector = array();
foreach ( $users as $user ) {
$user_selector[] = array(
'id' => $user->ID,
'name' => $user->display_name . ' ( ' . $user->user_email . ' )',
);
}
$bwfan_settings['bwf_notification_user_selector'] = $user_selector;
/** Set scheduler */
$old_settings = array(
'bwfan_notification_time' => array(
'hours' => '09',
'minutes' => '00',
'ampm' => 'am',
),
);
self::$instance->set_scheduler( $old_settings, $new_settings );
update_option( 'bwfan_global_settings', $bwfan_settings );
}
}
if ( class_exists( 'BWFAN_Core' ) ) {
BWFAN_Core::register( 'notification_email', 'BWFAN_Notification_Email' );
}