| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659 |
- <?php
- /**
- * Transactional Emails Controller
- *
- * WooCommerce Emails Class which handles the sending on transactional emails and email templates. This class loads in available emails.
- *
- * @package WooCommerce/Classes/Emails
- * @version 2.3.0
- */
- defined( 'ABSPATH' ) || exit;
- /**
- * Emails class.
- */
- class WC_Emails {
- /**
- * Array of email notification classes
- *
- * @var array
- */
- public $emails = array();
- /**
- * The single instance of the class
- *
- * @var WC_Emails
- */
- protected static $_instance = null;
- /**
- * Background emailer class.
- *
- * @var WC_Background_Emailer
- */
- protected static $background_emailer = null;
- /**
- * Main WC_Emails Instance.
- *
- * Ensures only one instance of WC_Emails is loaded or can be loaded.
- *
- * @since 2.1
- * @static
- * @return WC_Emails Main instance
- */
- public static function instance() {
- if ( is_null( self::$_instance ) ) {
- self::$_instance = new self();
- }
- return self::$_instance;
- }
- /**
- * Cloning is forbidden.
- *
- * @since 2.1
- */
- public function __clone() {
- wc_doing_it_wrong( __FUNCTION__, __( 'Cloning is forbidden.', 'woocommerce' ), '2.1' );
- }
- /**
- * Unserializing instances of this class is forbidden.
- *
- * @since 2.1
- */
- public function __wakeup() {
- wc_doing_it_wrong( __FUNCTION__, __( 'Unserializing instances of this class is forbidden.', 'woocommerce' ), '2.1' );
- }
- /**
- * Hook in all transactional emails.
- */
- public static function init_transactional_emails() {
- $email_actions = apply_filters(
- 'woocommerce_email_actions', array(
- 'woocommerce_low_stock',
- 'woocommerce_no_stock',
- 'woocommerce_product_on_backorder',
- 'woocommerce_order_status_pending_to_processing',
- 'woocommerce_order_status_pending_to_completed',
- 'woocommerce_order_status_processing_to_cancelled',
- 'woocommerce_order_status_pending_to_failed',
- 'woocommerce_order_status_pending_to_on-hold',
- 'woocommerce_order_status_failed_to_processing',
- 'woocommerce_order_status_failed_to_completed',
- 'woocommerce_order_status_failed_to_on-hold',
- 'woocommerce_order_status_on-hold_to_processing',
- 'woocommerce_order_status_on-hold_to_cancelled',
- 'woocommerce_order_status_on-hold_to_failed',
- 'woocommerce_order_status_completed',
- 'woocommerce_order_fully_refunded',
- 'woocommerce_order_partially_refunded',
- 'woocommerce_new_customer_note',
- 'woocommerce_created_customer',
- )
- );
- if ( apply_filters( 'woocommerce_defer_transactional_emails', false ) ) {
- self::$background_emailer = new WC_Background_Emailer();
- foreach ( $email_actions as $action ) {
- add_action( $action, array( __CLASS__, 'queue_transactional_email' ), 10, 10 );
- }
- } else {
- foreach ( $email_actions as $action ) {
- add_action( $action, array( __CLASS__, 'send_transactional_email' ), 10, 10 );
- }
- }
- }
- /**
- * Queues transactional email so it's not sent in current request if enabled,
- * otherwise falls back to send now.
- */
- public static function queue_transactional_email() {
- if ( is_a( self::$background_emailer, 'WC_Background_Emailer' ) ) {
- self::$background_emailer->push_to_queue(
- array(
- 'filter' => current_filter(),
- 'args' => func_get_args(),
- )
- );
- } else {
- call_user_func_array( array( __CLASS__, 'send_transactional_email' ), func_get_args() );
- }
- }
- /**
- * Init the mailer instance and call the notifications for the current filter.
- *
- * @internal
- *
- * @param string $filter Filter name.
- * @param array $args Email args (default: []).
- */
- public static function send_queued_transactional_email( $filter = '', $args = array() ) {
- if ( apply_filters( 'woocommerce_allow_send_queued_transactional_email', true, $filter, $args ) ) {
- self::instance(); // Init self so emails exist.
- // Ensure gateways are loaded in case they need to insert data into the emails.
- WC()->payment_gateways();
- WC()->shipping();
- do_action_ref_array( $filter . '_notification', $args );
- }
- }
- /**
- * Init the mailer instance and call the notifications for the current filter.
- *
- * @internal
- *
- * @param array $args Email args (default: []).
- */
- public static function send_transactional_email( $args = array() ) {
- try {
- $args = func_get_args();
- self::instance(); // Init self so emails exist.
- do_action_ref_array( current_filter() . '_notification', $args );
- } catch ( Exception $e ) {
- if ( defined( 'WP_DEBUG' ) && WP_DEBUG ) {
- trigger_error( 'Transactional email triggered fatal error for callback ' . current_filter(), E_USER_WARNING ); // phpcs:ignore WordPress.XSS.EscapeOutput.OutputNotEscaped, WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
- }
- }
- }
- /**
- * Constructor for the email class hooks in all emails that can be sent.
- */
- public function __construct() {
- $this->init();
- // Email Header, Footer and content hooks.
- add_action( 'woocommerce_email_header', array( $this, 'email_header' ) );
- add_action( 'woocommerce_email_footer', array( $this, 'email_footer' ) );
- add_action( 'woocommerce_email_order_details', array( $this, 'order_downloads' ), 10, 4 );
- add_action( 'woocommerce_email_order_details', array( $this, 'order_details' ), 10, 4 );
- add_action( 'woocommerce_email_order_meta', array( $this, 'order_meta' ), 10, 3 );
- add_action( 'woocommerce_email_customer_details', array( $this, 'customer_details' ), 10, 3 );
- add_action( 'woocommerce_email_customer_details', array( $this, 'email_addresses' ), 20, 3 );
- // Hooks for sending emails during store events.
- add_action( 'woocommerce_low_stock_notification', array( $this, 'low_stock' ) );
- add_action( 'woocommerce_no_stock_notification', array( $this, 'no_stock' ) );
- add_action( 'woocommerce_product_on_backorder_notification', array( $this, 'backorder' ) );
- add_action( 'woocommerce_created_customer_notification', array( $this, 'customer_new_account' ), 10, 3 );
- // Hook for replacing {site_title} in email-footer.
- add_filter( 'woocommerce_email_footer_text', array( $this, 'email_footer_replace_site_title' ) );
- // Let 3rd parties unhook the above via this hook.
- do_action( 'woocommerce_email', $this );
- }
- /**
- * Init email classes.
- */
- public function init() {
- // Include email classes.
- include_once dirname( __FILE__ ) . '/emails/class-wc-email.php';
- $this->emails['WC_Email_New_Order'] = include 'emails/class-wc-email-new-order.php';
- $this->emails['WC_Email_Cancelled_Order'] = include 'emails/class-wc-email-cancelled-order.php';
- $this->emails['WC_Email_Failed_Order'] = include 'emails/class-wc-email-failed-order.php';
- $this->emails['WC_Email_Customer_On_Hold_Order'] = include 'emails/class-wc-email-customer-on-hold-order.php';
- $this->emails['WC_Email_Customer_Processing_Order'] = include 'emails/class-wc-email-customer-processing-order.php';
- $this->emails['WC_Email_Customer_Completed_Order'] = include 'emails/class-wc-email-customer-completed-order.php';
- $this->emails['WC_Email_Customer_Refunded_Order'] = include 'emails/class-wc-email-customer-refunded-order.php';
- $this->emails['WC_Email_Customer_Invoice'] = include 'emails/class-wc-email-customer-invoice.php';
- $this->emails['WC_Email_Customer_Note'] = include 'emails/class-wc-email-customer-note.php';
- $this->emails['WC_Email_Customer_Reset_Password'] = include 'emails/class-wc-email-customer-reset-password.php';
- $this->emails['WC_Email_Customer_New_Account'] = include 'emails/class-wc-email-customer-new-account.php';
- $this->emails = apply_filters( 'woocommerce_email_classes', $this->emails );
- // include css inliner.
- if ( ! class_exists( 'Emogrifier' ) && class_exists( 'DOMDocument' ) ) {
- include_once dirname( __FILE__ ) . '/libraries/class-emogrifier.php';
- }
- }
- /**
- * Return the email classes - used in admin to load settings.
- *
- * @return array
- */
- public function get_emails() {
- return $this->emails;
- }
- /**
- * Get from name for email.
- *
- * @return string
- */
- public function get_from_name() {
- return wp_specialchars_decode( get_option( 'woocommerce_email_from_name' ), ENT_QUOTES );
- }
- /**
- * Get from email address.
- *
- * @return string
- */
- public function get_from_address() {
- return sanitize_email( get_option( 'woocommerce_email_from_address' ) );
- }
- /**
- * Get the email header.
- *
- * @param mixed $email_heading Heading for the email.
- */
- public function email_header( $email_heading ) {
- wc_get_template( 'emails/email-header.php', array( 'email_heading' => $email_heading ) );
- }
- /**
- * Get the email footer.
- */
- public function email_footer() {
- wc_get_template( 'emails/email-footer.php' );
- }
- /**
- * Filter callback to replace {site_title} in email footer
- *
- * @since 3.3.0
- * @param string $string Email footer text.
- * @return string Email footer text with any replacements done.
- */
- public function email_footer_replace_site_title( $string ) {
- return str_replace( '{site_title}', $this->get_blogname(), $string );
- }
- /**
- * Wraps a message in the woocommerce mail template.
- *
- * @param string $email_heading Heading text.
- * @param string $message Email message.
- * @param bool $plain_text Set true to send as plain text. Default to false.
- *
- * @return string
- */
- public function wrap_message( $email_heading, $message, $plain_text = false ) {
- // Buffer.
- ob_start();
- do_action( 'woocommerce_email_header', $email_heading, null );
- echo wpautop( wptexturize( $message ) ); // WPCS: XSS ok.
- do_action( 'woocommerce_email_footer', null );
- // Get contents.
- $message = ob_get_clean();
- return $message;
- }
- /**
- * Send the email.
- *
- * @param mixed $to Receiver.
- * @param mixed $subject Email subject.
- * @param mixed $message Message.
- * @param string $headers Email headers (default: "Content-Type: text/html\r\n").
- * @param string $attachments Attachments (default: "").
- * @return bool
- */
- public function send( $to, $subject, $message, $headers = "Content-Type: text/html\r\n", $attachments = '' ) {
- // Send.
- $email = new WC_Email();
- return $email->send( $to, $subject, $message, $headers, $attachments );
- }
- /**
- * Prepare and send the customer invoice email on demand.
- *
- * @param int|WC_Order $order Order instance or ID.
- */
- public function customer_invoice( $order ) {
- $email = $this->emails['WC_Email_Customer_Invoice'];
- if ( ! is_object( $order ) ) {
- $order = wc_get_order( absint( $order ) );
- }
- $email->trigger( $order->get_id(), $order );
- }
- /**
- * Customer new account welcome email.
- *
- * @param int $customer_id Customer ID.
- * @param array $new_customer_data New customer data.
- * @param bool $password_generated If password is generated.
- */
- public function customer_new_account( $customer_id, $new_customer_data = array(), $password_generated = false ) {
- if ( ! $customer_id ) {
- return;
- }
- $user_pass = ! empty( $new_customer_data['user_pass'] ) ? $new_customer_data['user_pass'] : '';
- $email = $this->emails['WC_Email_Customer_New_Account'];
- $email->trigger( $customer_id, $user_pass, $password_generated );
- }
- /**
- * Show the order details table
- *
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- * @param string $email Email address.
- */
- public function order_details( $order, $sent_to_admin = false, $plain_text = false, $email = '' ) {
- if ( $plain_text ) {
- wc_get_template(
- 'emails/plain/email-order-details.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- 'plain_text' => $plain_text,
- 'email' => $email,
- )
- );
- } else {
- wc_get_template(
- 'emails/email-order-details.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- 'plain_text' => $plain_text,
- 'email' => $email,
- )
- );
- }
- }
- /**
- * Show order downloads in a table.
- *
- * @since 3.2.0
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- * @param string $email Email address.
- */
- public function order_downloads( $order, $sent_to_admin = false, $plain_text = false, $email = '' ) {
- $show_downloads = $order->has_downloadable_item() && $order->is_download_permitted() && ! $sent_to_admin;
- if ( ! $show_downloads ) {
- return;
- }
- $downloads = $order->get_downloadable_items();
- $columns = apply_filters(
- 'woocommerce_email_downloads_columns', array(
- 'download-product' => __( 'Product', 'woocommerce' ),
- 'download-expires' => __( 'Expires', 'woocommerce' ),
- 'download-file' => __( 'Download', 'woocommerce' ),
- )
- );
- if ( $plain_text ) {
- wc_get_template(
- 'emails/plain/email-downloads.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- 'plain_text' => $plain_text,
- 'email' => $email,
- 'downloads' => $downloads,
- 'columns' => $columns,
- )
- );
- } else {
- wc_get_template(
- 'emails/email-downloads.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- 'plain_text' => $plain_text,
- 'email' => $email,
- 'downloads' => $downloads,
- 'columns' => $columns,
- )
- );
- }
- }
- /**
- * Add order meta to email templates.
- *
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- */
- public function order_meta( $order, $sent_to_admin = false, $plain_text = false ) {
- $fields = apply_filters( 'woocommerce_email_order_meta_fields', array(), $sent_to_admin, $order );
- /**
- * Deprecated woocommerce_email_order_meta_keys filter.
- *
- * @since 2.3.0
- */
- $_fields = apply_filters( 'woocommerce_email_order_meta_keys', array(), $sent_to_admin );
- if ( $_fields ) {
- foreach ( $_fields as $key => $field ) {
- if ( is_numeric( $key ) ) {
- $key = $field;
- }
- $fields[ $key ] = array(
- 'label' => wptexturize( $key ),
- 'value' => wptexturize( get_post_meta( $order->get_id(), $field, true ) ),
- );
- }
- }
- if ( $fields ) {
- if ( $plain_text ) {
- foreach ( $fields as $field ) {
- if ( isset( $field['label'] ) && isset( $field['value'] ) && $field['value'] ) {
- echo $field['label'] . ': ' . $field['value'] . "\n"; // WPCS: XSS ok.
- }
- }
- } else {
- foreach ( $fields as $field ) {
- if ( isset( $field['label'] ) && isset( $field['value'] ) && $field['value'] ) {
- echo '<p><strong>' . $field['label'] . ':</strong> ' . $field['value'] . '</p>'; // WPCS: XSS ok.
- }
- }
- }
- }
- }
- /**
- * Is customer detail field valid?
- *
- * @param array $field Field data to check if is valid.
- * @return boolean
- */
- public function customer_detail_field_is_valid( $field ) {
- return isset( $field['label'] ) && ! empty( $field['value'] );
- }
- /**
- * Allows developers to add additional customer details to templates.
- *
- * In versions prior to 3.2 this was used for notes, phone and email but this data has moved.
- *
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- */
- public function customer_details( $order, $sent_to_admin = false, $plain_text = false ) {
- if ( ! is_a( $order, 'WC_Order' ) ) {
- return;
- }
- $fields = array_filter( apply_filters( 'woocommerce_email_customer_details_fields', array(), $sent_to_admin, $order ), array( $this, 'customer_detail_field_is_valid' ) );
- if ( ! empty( $fields ) ) {
- if ( $plain_text ) {
- wc_get_template( 'emails/plain/email-customer-details.php', array( 'fields' => $fields ) );
- } else {
- wc_get_template( 'emails/email-customer-details.php', array( 'fields' => $fields ) );
- }
- }
- }
- /**
- * Get the email addresses.
- *
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- */
- public function email_addresses( $order, $sent_to_admin = false, $plain_text = false ) {
- if ( ! is_a( $order, 'WC_Order' ) ) {
- return;
- }
- if ( $plain_text ) {
- wc_get_template(
- 'emails/plain/email-addresses.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- )
- );
- } else {
- wc_get_template(
- 'emails/email-addresses.php', array(
- 'order' => $order,
- 'sent_to_admin' => $sent_to_admin,
- )
- );
- }
- }
- /**
- * Get blog name formatted for emails.
- *
- * @return string
- */
- private function get_blogname() {
- return wp_specialchars_decode( get_option( 'blogname' ), ENT_QUOTES );
- }
- /**
- * Low stock notification email.
- *
- * @param WC_Product $product Product instance.
- */
- public function low_stock( $product ) {
- if ( 'no' === get_option( 'woocommerce_notify_low_stock', 'yes' ) ) {
- return;
- }
- $subject = sprintf( '[%s] %s', $this->get_blogname(), __( 'Product low in stock', 'woocommerce' ) );
- $message = sprintf(
- /* translators: 1: product name 2: items in stock */
- __( '%1$s is low in stock. There are %2$d left.', 'woocommerce' ),
- html_entity_decode( strip_tags( $product->get_formatted_name() ), ENT_QUOTES, get_bloginfo( 'charset' ) ),
- html_entity_decode( strip_tags( $product->get_stock_quantity() ) )
- );
- wp_mail(
- apply_filters( 'woocommerce_email_recipient_low_stock', get_option( 'woocommerce_stock_email_recipient' ), $product ),
- apply_filters( 'woocommerce_email_subject_low_stock', $subject, $product ),
- apply_filters( 'woocommerce_email_content_low_stock', $message, $product ),
- apply_filters( 'woocommerce_email_headers', '', 'low_stock', $product ),
- apply_filters( 'woocommerce_email_attachments', array(), 'low_stock', $product )
- );
- }
- /**
- * No stock notification email.
- *
- * @param WC_Product $product Product instance.
- */
- public function no_stock( $product ) {
- if ( 'no' === get_option( 'woocommerce_notify_no_stock', 'yes' ) ) {
- return;
- }
- $subject = sprintf( '[%s] %s', $this->get_blogname(), __( 'Product out of stock', 'woocommerce' ) );
- /* translators: %s: product name */
- $message = sprintf( __( '%s is out of stock.', 'woocommerce' ), html_entity_decode( strip_tags( $product->get_formatted_name() ), ENT_QUOTES, get_bloginfo( 'charset' ) ) );
- wp_mail(
- apply_filters( 'woocommerce_email_recipient_no_stock', get_option( 'woocommerce_stock_email_recipient' ), $product ),
- apply_filters( 'woocommerce_email_subject_no_stock', $subject, $product ),
- apply_filters( 'woocommerce_email_content_no_stock', $message, $product ),
- apply_filters( 'woocommerce_email_headers', '', 'no_stock', $product ),
- apply_filters( 'woocommerce_email_attachments', array(), 'no_stock', $product )
- );
- }
- /**
- * Backorder notification email.
- *
- * @param array $args Arguments.
- */
- public function backorder( $args ) {
- $args = wp_parse_args(
- $args, array(
- 'product' => '',
- 'quantity' => '',
- 'order_id' => '',
- )
- );
- $order = wc_get_order( $args['order_id'] );
- if (
- ! $args['product'] ||
- ! is_object( $args['product'] ) ||
- ! $args['quantity'] ||
- ! $order
- ) {
- return;
- }
- $subject = sprintf( '[%s] %s', $this->get_blogname(), __( 'Product backorder', 'woocommerce' ) );
- /* translators: 1: product quantity 2: product name 3: order number */
- $message = sprintf( __( '%1$s units of %2$s have been backordered in order #%3$s.', 'woocommerce' ), $args['quantity'], html_entity_decode( strip_tags( $args['product']->get_formatted_name() ), ENT_QUOTES, get_bloginfo( 'charset' ) ), $order->get_order_number() );
- wp_mail(
- apply_filters( 'woocommerce_email_recipient_backorder', get_option( 'woocommerce_stock_email_recipient' ), $args ),
- apply_filters( 'woocommerce_email_subject_backorder', $subject, $args ),
- apply_filters( 'woocommerce_email_content_backorder', $message, $args ),
- apply_filters( 'woocommerce_email_headers', '', 'backorder', $args ),
- apply_filters( 'woocommerce_email_attachments', array(), 'backorder', $args )
- );
- }
- /**
- * Adds Schema.org markup for order in JSON-LD format.
- *
- * @deprecated 3.0.0
- * @see WC_Structured_Data::generate_order_data()
- *
- * @since 2.6.0
- * @param WC_Order $order Order instance.
- * @param bool $sent_to_admin If should sent to admin.
- * @param bool $plain_text If is plain text email.
- */
- public function order_schema_markup( $order, $sent_to_admin = false, $plain_text = false ) {
- wc_deprecated_function( 'WC_Emails::order_schema_markup', '3.0', 'WC_Structured_Data::generate_order_data' );
- WC()->structured_data->generate_order_data( $order, $sent_to_admin, $plain_text );
- WC()->structured_data->output_structured_data();
- }
- }
|