File "EmailPreview-20250317144851.php"
Full Path: /home/amervokv/ecomlive.net/wp-content/plugins/woocommerce/src/Internal/Admin/EmailPreview/EmailPreview-20250317144851.php
File size: 13.34 KB
MIME-type: text/x-php
Charset: utf-8
<?php
/**
* Renders the email preview.
*/
declare( strict_types=1 );
namespace Automattic\WooCommerce\Internal\Admin\EmailPreview;
use Automattic\WooCommerce\Utilities\FeaturesUtil;
use WC_Email;
use WC_Order;
use WC_Product;
use WC_Product_Variation;
use WP_User;
defined( 'ABSPATH' ) || exit;
/**
* EmailPreview Class.
*/
class EmailPreview {
const DEFAULT_EMAIL_TYPE = 'WC_Email_Customer_Processing_Order';
const DEFAULT_EMAIL_ID = 'customer_processing_order';
const USER_OBJECT_EMAILS = array(
'WC_Email_Customer_New_Account',
'WC_Email_Customer_Reset_Password',
);
/**
* All fields IDs that can customize email styles in Settings.
*
* @var array
*/
private static array $email_style_settings_ids = array(
'woocommerce_email_background_color',
'woocommerce_email_base_color',
'woocommerce_email_body_background_color',
'woocommerce_email_font_family',
'woocommerce_email_footer_text',
'woocommerce_email_footer_text_color',
'woocommerce_email_header_alignment',
'woocommerce_email_header_image',
'woocommerce_email_text_color',
);
/**
* All fields IDs that can customize specific email content in Settings.
*
* @var array
*/
private static array $email_content_settings_ids = array();
/**
* Whether the email settings IDs are initialized.
*
* @var bool
*/
private static bool $email_settings_ids_initialized = false;
/**
* The email type to preview.
*
* @var string|null
*/
private ?string $email_type = null;
/**
* The email object.
*
* @var WC_Email|null
*/
private ?WC_Email $email = null;
/**
* The single instance of the class.
*
* @var object
*/
protected static $instance = null;
/**
* Get class instance.
*
* @return object Instance.
*/
final public static function instance() {
if ( null === static::$instance ) {
static::$instance = new static();
}
return static::$instance;
}
/**
* Get all email settings IDs.
*/
public static function get_all_email_settings_ids() {
if ( ! self::$email_settings_ids_initialized ) {
self::$email_settings_ids_initialized = true;
$emails = WC()->mailer()->get_emails();
foreach ( $emails as $email ) {
self::$email_content_settings_ids = array_merge(
self::$email_content_settings_ids,
self::get_email_content_settings_ids( $email->id )
);
}
self::$email_content_settings_ids = array_unique( self::$email_content_settings_ids );
}
return array_merge(
self::$email_style_settings_ids,
self::$email_content_settings_ids,
);
}
/**
* Get email style settings IDs.
*/
public static function get_email_style_settings_ids() {
return self::$email_style_settings_ids;
}
/**
* Get email content settings IDs for specific email.
*
* @param string|null $email_id Email ID.
*/
public static function get_email_content_settings_ids( ?string $email_id ) {
if ( ! $email_id ) {
return array();
}
return array(
"woocommerce_{$email_id}_subject",
"woocommerce_{$email_id}_heading",
"woocommerce_{$email_id}_additional_content",
);
}
/**
* Set the email type to preview.
*
* @param string $email_type Email type.
*
* @throws \InvalidArgumentException When the email type is invalid.
*/
public function set_email_type( string $email_type ) {
$emails = WC()->mailer()->get_emails();
if ( ! in_array( $email_type, array_keys( $emails ), true ) ) {
throw new \InvalidArgumentException( 'Invalid email type' );
}
$this->email_type = $email_type;
$this->email = $emails[ $email_type ];
$object = null;
if ( in_array( $email_type, self::USER_OBJECT_EMAILS, true ) ) {
$object = new WP_User( 0 );
$this->email->user_email = 'user_preview@example.com';
$this->email->user_login = 'user_preview';
$this->email->set_object( $object );
} else {
$object = $this->get_dummy_order();
if ( $email_type === 'WC_Email_Customer_Note' ) {
$this->email->customer_note = $object->get_customer_note();
}
if ( $email_type === 'WC_Email_Customer_Refunded_Order' ) {
$this->email->partial_refund = false;
}
$this->email->set_object( $object );
}
$this->email->placeholders = array_merge(
$this->email->placeholders,
$this->get_placeholders( $object )
);
/**
* Allow to modify the email object before rendering the preview to add additional data.
*
* @param WC_Email $email The email object.
*
* @since 9.6.0
*/
$this->email = apply_filters( 'woocommerce_prepare_email_for_preview', $this->email );
}
/**
* Get the preview email content.
*
* @return string
*/
public function render() {
if ( FeaturesUtil::feature_is_enabled( 'email_improvements' ) ) {
return $this->render_preview_email();
}
return $this->render_legacy_preview_email();
}
/**
* Ensure links open in new tab. User in WooCommerce Settings,
* so the links don't open inside the iframe.
*
* @param string $content Email content HTML.
* @return string
*/
public function ensure_links_open_in_new_tab( string $content ) {
return (string) preg_replace_callback(
'/<a\s+([^>]*?)(target=["\']?[^"\']*["\']?)?([^>]*)>/i',
function ( $matches ) {
$before = $matches[1];
$target = 'target="_blank"';
$after = $matches[3];
return "<a $before $target $after>";
},
$content
);
}
/**
* Get the preview email content.
*
* @return string
*/
public function get_subject() {
if ( ! $this->email ) {
return '';
}
$this->set_up_filters();
$subject = $this->email->get_subject();
$this->clean_up_filters();
return $subject;
}
/**
* Return a dummy product when the product is not set in email classes.
*
* @param WC_Product|null $product Order item product.
* @return WC_Product
*/
public function get_dummy_product_when_not_set( $product ) {
if ( $product ) {
return $product;
}
return $this->get_dummy_product();
}
/**
* Get HTML of the legacy preview email.
*
* @return string
*/
private function render_legacy_preview_email() {
// load the mailer class.
$mailer = WC()->mailer();
// get the preview email subject.
$email_heading = __( 'HTML email template', 'woocommerce' );
// get the preview email content.
ob_start();
include WC()->plugin_path() . '/includes/admin/views/html-email-template-preview.php';
$message = ob_get_clean();
// create a new email.
$email = new WC_Email();
/**
* Wrap the content with the email template and then add styles.
*
* @since 2.6.0
*/
return apply_filters( 'woocommerce_mail_content', $email->style_inline( $mailer->wrap_message( $email_heading, $message ) ) );
}
/**
* Render HTML content of the preview email.
*
* @return string
*/
private function render_preview_email() {
$this->set_up_filters();
if ( ! $this->email_type ) {
$this->set_email_type( self::DEFAULT_EMAIL_TYPE );
}
$content = $this->email->get_content_html();
$inlined = $this->email->style_inline( $content );
$this->clean_up_filters();
/** This filter is documented in src/Internal/Admin/EmailPreview/EmailPreview.php */
return apply_filters( 'woocommerce_mail_content', $inlined ); // phpcs:ignore WooCommerce.Commenting.CommentHooks.MissingSinceComment
}
/**
* Get a dummy order object without the need to create in the database.
*
* @return WC_Order
*/
private function get_dummy_order() {
$product = $this->get_dummy_product();
$variation = $this->get_dummy_product_variation();
$order = new WC_Order();
if ( $product ) {
$order->add_product( $product, 2 );
}
if ( $variation ) {
$order->add_product( $variation );
}
$order->set_id( 12345 );
$order->set_date_created( time() );
$order->set_currency( 'USD' );
$order->set_discount_total( 10 );
$order->set_shipping_total( 5 );
$order->set_total( 65 );
$order->set_payment_method_title( __( 'Direct bank transfer', 'woocommerce' ) );
$order->set_customer_note( __( "This is a customer note. Customers can add a note to their order on checkout.\n\nIt can be multiple lines. If there’s no note, this section is hidden.", 'woocommerce' ) );
$address = $this->get_dummy_address();
$order->set_billing_address( $address );
$order->set_shipping_address( $address );
/**
* A dummy WC_Order used in email preview.
*
* @param WC_Order $order The dummy order object.
* @param string $email_type The email type to preview.
*
* @since 9.6.0
*/
return apply_filters( 'woocommerce_email_preview_dummy_order', $order, $this->email_type );
}
/**
* Get a dummy product. Also used with `woocommerce_order_item_product` filter
* when email templates tries to get the product from the database.
*
* @return WC_Product
*/
private function get_dummy_product() {
$product = new WC_Product();
$product->set_name( 'Dummy Product' );
$product->set_price( 25 );
/**
* A dummy WC_Product used in email preview.
*
* @param WC_Product $product The dummy product object.
* @param string $email_type The email type to preview.
*
* @since 9.6.0
*/
return apply_filters( 'woocommerce_email_preview_dummy_product', $product, $this->email_type );
}
/**
* Get a dummy product variation.
*
* @return WC_Product_Variation
*/
private function get_dummy_product_variation() {
$variation = new WC_Product_Variation();
$variation->set_name( 'Dummy Product Variation' );
$variation->set_price( 20 );
$variation->set_attributes(
array(
'pa_color' => 'red',
'pa_size' => 'small',
)
);
/**
* A dummy WC_Product_Variation used in email preview.
*
* @param WC_Product_Variation $variation The dummy product variation object.
* @param string $email_type The email type to preview.
*
* @since 9.7.0
*/
return apply_filters( 'woocommerce_email_preview_dummy_product_variation', $variation, $this->email_type );
}
/**
* Get a dummy address.
*
* @return array
*/
private function get_dummy_address() {
$address = array(
'first_name' => 'John',
'last_name' => 'Doe',
'company' => 'Company',
'email' => 'john@company.com',
'phone' => '555-555-5555',
'address_1' => '123 Fake Street',
'city' => 'Faketown',
'postcode' => '12345',
'country' => 'US',
'state' => 'CA',
);
/**
* A dummy address used in email preview as billing and shipping one.
*
* @param array $address The dummy address.
* @param string $email_type The email type to preview.
*
* @since 9.6.0
*/
return apply_filters( 'woocommerce_email_preview_dummy_address', $address, $this->email_type );
}
/**
* Get the placeholders for the email preview.
*
* @param WC_Order|WP_User $email_object The object to render email with.
* @return array
*/
private function get_placeholders( $email_object ) {
$placeholders = array();
if ( is_a( $email_object, 'WC_Order' ) ) {
$placeholders['{order_date}'] = wc_format_datetime( $email_object->get_date_created() );
$placeholders['{order_number}'] = $email_object->get_order_number();
$placeholders['{order_billing_full_name}'] = $email_object->get_formatted_billing_full_name();
}
/**
* Placeholders for email preview.
*
* @param WC_Order $placeholders Placeholders for email subject.
* @param string $email_type The email type to preview.
*
* @since 9.6.0
*/
return apply_filters( 'woocommerce_email_preview_placeholders', $placeholders, $this->email_type );
}
/**
* Set up filters for email preview.
*/
private function set_up_filters() {
// Always show shipping address in the preview email.
add_filter( 'woocommerce_order_needs_shipping_address', array( $this, 'enable_shipping_address' ) );
// Email templates fetch product from the database to show additional information, which are not
// saved in WC_Order_Item_Product. This filter enables fetching that data also in email preview.
add_filter( 'woocommerce_order_item_product', array( $this, 'get_dummy_product_when_not_set' ), 10, 1 );
// Enable email preview mode - this way transient values are fetched for live preview.
add_filter( 'woocommerce_is_email_preview', array( $this, 'enable_preview_mode' ) );
// Get shipping method without needing to save it in the order.
add_filter( 'woocommerce_order_shipping_method', array( $this, 'get_shipping_method' ) );
}
/**
* Clean up filters after email preview.
*/
private function clean_up_filters() {
remove_filter( 'woocommerce_order_needs_shipping_address', array( $this, 'enable_shipping_address' ) );
remove_filter( 'woocommerce_order_item_product', array( $this, 'get_dummy_product_when_not_set' ), 10 );
remove_filter( 'woocommerce_is_email_preview', array( $this, 'enable_preview_mode' ) );
remove_filter( 'woocommerce_order_shipping_method', array( $this, 'get_shipping_method' ) );
}
/**
* Get the shipping method for the preview email.
*
* @return string
*/
public function get_shipping_method() {
return __( 'Flat rate', 'woocommerce' );
}
/**
* Enable shipping address in the preview email. Not using __return_true so
* we don't accidentally remove the same filter used by other plugin or theme.
*
* @return true
*/
public function enable_shipping_address() {
return true;
}
/**
* Enable preview mode to use transient values in email-styles.php. Not using __return_true
* so we don't accidentally remove the same filter used by other plugin or theme.
*
* @return true
*/
public function enable_preview_mode() {
return true;
}
}