Fix button roundtrip in editor, alignment persistence, and test email rendering
This commit is contained in:
@@ -226,6 +226,26 @@ class NotificationsController {
|
||||
'permission_callback' => [$this, 'check_permission'],
|
||||
],
|
||||
]);
|
||||
|
||||
// POST /woonoow/v1/notifications/templates/:eventId/:channelId/send-test
|
||||
register_rest_route($this->namespace, '/' . $this->rest_base . '/templates/(?P<eventId>[a-zA-Z0-9_-]+)/(?P<channelId>[a-zA-Z0-9_-]+)/send-test', [
|
||||
[
|
||||
'methods' => 'POST',
|
||||
'callback' => [$this, 'send_test_email'],
|
||||
'permission_callback' => [$this, 'check_permission'],
|
||||
'args' => [
|
||||
'email' => [
|
||||
'required' => true,
|
||||
'type' => 'string',
|
||||
'sanitize_callback' => 'sanitize_email',
|
||||
],
|
||||
'recipient' => [
|
||||
'default' => 'customer',
|
||||
'type' => 'string',
|
||||
],
|
||||
],
|
||||
],
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -931,4 +951,411 @@ class NotificationsController {
|
||||
'per_page' => $per_page,
|
||||
], 200);
|
||||
}
|
||||
|
||||
/**
|
||||
* Send test email for a notification template
|
||||
*
|
||||
* @param WP_REST_Request $request Request object
|
||||
* @return WP_REST_Response|WP_Error
|
||||
*/
|
||||
public function send_test_email(WP_REST_Request $request) {
|
||||
$event_id = $request->get_param('eventId');
|
||||
$channel_id = $request->get_param('channelId');
|
||||
$recipient_type = $request->get_param('recipient') ?? 'customer';
|
||||
$to_email = $request->get_param('email');
|
||||
|
||||
// Validate email
|
||||
if (!is_email($to_email)) {
|
||||
return new \WP_Error(
|
||||
'invalid_email',
|
||||
__('Invalid email address', 'woonoow'),
|
||||
['status' => 400]
|
||||
);
|
||||
}
|
||||
|
||||
// Only support email channel for test
|
||||
if ($channel_id !== 'email') {
|
||||
return new \WP_Error(
|
||||
'unsupported_channel',
|
||||
__('Test sending is only available for email channel', 'woonoow'),
|
||||
['status' => 400]
|
||||
);
|
||||
}
|
||||
|
||||
// Get template
|
||||
$template = TemplateProvider::get_template($event_id, $channel_id, $recipient_type);
|
||||
|
||||
if (!$template) {
|
||||
return new \WP_Error(
|
||||
'template_not_found',
|
||||
__('Template not found', 'woonoow'),
|
||||
['status' => 404]
|
||||
);
|
||||
}
|
||||
|
||||
// Build sample data for variables
|
||||
$sample_data = $this->get_sample_data_for_event($event_id);
|
||||
|
||||
// Replace variables in subject and body
|
||||
$subject = '[TEST] ' . $this->replace_variables($template['subject'] ?? '', $sample_data);
|
||||
$body_markdown = $this->replace_variables($template['body'] ?? '', $sample_data);
|
||||
|
||||
// Render email using EmailRenderer
|
||||
$email_renderer = \WooNooW\Core\Notifications\EmailRenderer::instance();
|
||||
|
||||
// We need to manually render since we're not triggering a real event
|
||||
$html = $this->render_test_email($body_markdown, $subject, $sample_data);
|
||||
|
||||
// Set content type to HTML
|
||||
$headers = ['Content-Type: text/html; charset=UTF-8'];
|
||||
|
||||
// Send email
|
||||
$sent = wp_mail($to_email, $subject, $html, $headers);
|
||||
|
||||
if (!$sent) {
|
||||
return new \WP_Error(
|
||||
'send_failed',
|
||||
__('Failed to send test email. Check your mail server configuration.', 'woonoow'),
|
||||
['status' => 500]
|
||||
);
|
||||
}
|
||||
|
||||
return new WP_REST_Response([
|
||||
'success' => true,
|
||||
'message' => sprintf(__('Test email sent to %s', 'woonoow'), $to_email),
|
||||
], 200);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get sample data for an event type
|
||||
*
|
||||
* @param string $event_id
|
||||
* @return array
|
||||
*/
|
||||
private function get_sample_data_for_event($event_id) {
|
||||
$base_data = [
|
||||
'site_name' => get_bloginfo('name'),
|
||||
'store_name' => get_bloginfo('name'),
|
||||
'store_url' => home_url(),
|
||||
'shop_url' => get_permalink(wc_get_page_id('shop')),
|
||||
'my_account_url' => get_permalink(wc_get_page_id('myaccount')),
|
||||
'support_email' => get_option('admin_email'),
|
||||
'current_year' => date('Y'),
|
||||
'customer_name' => 'John Doe',
|
||||
'customer_first_name' => 'John',
|
||||
'customer_last_name' => 'Doe',
|
||||
'customer_email' => 'john@example.com',
|
||||
'customer_phone' => '+1 234 567 8900',
|
||||
'login_url' => wp_login_url(),
|
||||
];
|
||||
|
||||
// Order-related events
|
||||
if (strpos($event_id, 'order') !== false) {
|
||||
$base_data = array_merge($base_data, [
|
||||
'order_number' => '12345',
|
||||
'order_id' => '12345',
|
||||
'order_date' => date('F j, Y'),
|
||||
'order_total' => wc_price(129.99),
|
||||
'order_subtotal' => wc_price(109.99),
|
||||
'order_tax' => wc_price(10.00),
|
||||
'order_shipping' => wc_price(10.00),
|
||||
'order_discount' => wc_price(0),
|
||||
'order_status' => 'Processing',
|
||||
'order_url' => '#',
|
||||
'payment_method' => 'Credit Card',
|
||||
'payment_status' => 'Paid',
|
||||
'payment_date' => date('F j, Y'),
|
||||
'transaction_id' => 'TXN123456789',
|
||||
'shipping_method' => 'Standard Shipping',
|
||||
'estimated_delivery' => date('F j', strtotime('+3 days')) . '-' . date('j', strtotime('+5 days')),
|
||||
'completion_date' => date('F j, Y'),
|
||||
'billing_address' => '123 Main St, City, State 12345, Country',
|
||||
'shipping_address' => '123 Main St, City, State 12345, Country',
|
||||
'tracking_number' => 'TRACK123456',
|
||||
'tracking_url' => '#',
|
||||
'shipping_carrier' => 'Standard Carrier',
|
||||
'payment_retry_url' => '#',
|
||||
'review_url' => '#',
|
||||
'order_items' => $this->get_sample_order_items_html(),
|
||||
'order_items_table' => $this->get_sample_order_items_html(),
|
||||
]);
|
||||
}
|
||||
|
||||
// Customer account events
|
||||
if (strpos($event_id, 'customer') !== false || strpos($event_id, 'account') !== false) {
|
||||
$base_data = array_merge($base_data, [
|
||||
'customer_username' => 'johndoe',
|
||||
'user_temp_password' => 'SamplePass123',
|
||||
'reset_link' => '#',
|
||||
'reset_key' => 'abc123xyz',
|
||||
'user_login' => 'johndoe',
|
||||
'user_email' => 'john@example.com',
|
||||
]);
|
||||
}
|
||||
|
||||
return $base_data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get sample order items HTML
|
||||
*
|
||||
* @return string
|
||||
*/
|
||||
private function get_sample_order_items_html() {
|
||||
return '<table style="width: 100%; border-collapse: collapse; margin: 16px 0;">
|
||||
<thead>
|
||||
<tr style="background: #f5f5f5;">
|
||||
<th style="padding: 12px; text-align: left; border-bottom: 2px solid #ddd;">Product</th>
|
||||
<th style="padding: 12px; text-align: center; border-bottom: 2px solid #ddd;">Qty</th>
|
||||
<th style="padding: 12px; text-align: right; border-bottom: 2px solid #ddd;">Price</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<td style="padding: 12px; border-bottom: 1px solid #eee;">
|
||||
<strong>Sample Product</strong><br>
|
||||
<span style="color: #666; font-size: 13px;">Size: M, Color: Blue</span>
|
||||
</td>
|
||||
<td style="padding: 12px; text-align: center; border-bottom: 1px solid #eee;">2</td>
|
||||
<td style="padding: 12px; text-align: right; border-bottom: 1px solid #eee;">' . wc_price(59.98) . '</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style="padding: 12px; border-bottom: 1px solid #eee;">
|
||||
<strong>Another Product</strong><br>
|
||||
<span style="color: #666; font-size: 13px;">Option: Standard</span>
|
||||
</td>
|
||||
<td style="padding: 12px; text-align: center; border-bottom: 1px solid #eee;">1</td>
|
||||
<td style="padding: 12px; text-align: right; border-bottom: 1px solid #eee;">' . wc_price(50.01) . '</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>';
|
||||
}
|
||||
|
||||
/**
|
||||
* Replace variables in text
|
||||
*
|
||||
* @param string $text
|
||||
* @param array $variables
|
||||
* @return string
|
||||
*/
|
||||
private function replace_variables($text, $variables) {
|
||||
foreach ($variables as $key => $value) {
|
||||
$text = str_replace('{' . $key . '}', $value, $text);
|
||||
}
|
||||
return $text;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render test email HTML
|
||||
*
|
||||
* @param string $body_markdown
|
||||
* @param string $subject
|
||||
* @param array $variables
|
||||
* @return string
|
||||
*/
|
||||
private function render_test_email($body_markdown, $subject, $variables) {
|
||||
// Parse cards
|
||||
$content = $this->parse_cards_for_test($body_markdown);
|
||||
|
||||
// Get appearance settings for colors
|
||||
$appearance = get_option('woonoow_appearance_settings', []);
|
||||
$colors = $appearance['general']['colors'] ?? [];
|
||||
$primary_color = $colors['primary'] ?? '#7f54b3';
|
||||
$secondary_color = $colors['secondary'] ?? '#7f54b3';
|
||||
$hero_gradient_start = $colors['gradientStart'] ?? '#667eea';
|
||||
$hero_gradient_end = $colors['gradientEnd'] ?? '#764ba2';
|
||||
|
||||
// Get email settings for branding
|
||||
$email_settings = get_option('woonoow_email_settings', []);
|
||||
$logo_url = $email_settings['logo_url'] ?? '';
|
||||
$header_text = $email_settings['header_text'] ?? $variables['store_name'];
|
||||
$footer_text = $email_settings['footer_text'] ?? sprintf('© %s %s. All rights reserved.', date('Y'), $variables['store_name']);
|
||||
$footer_text = str_replace('{current_year}', date('Y'), $footer_text);
|
||||
|
||||
// Build header
|
||||
if (!empty($logo_url)) {
|
||||
$header = sprintf(
|
||||
'<a href="%s"><img src="%s" alt="%s" style="max-width: 200px; max-height: 60px;"></a>',
|
||||
esc_url($variables['store_url']),
|
||||
esc_url($logo_url),
|
||||
esc_attr($variables['store_name'])
|
||||
);
|
||||
} else {
|
||||
$header = sprintf(
|
||||
'<a href="%s" style="font-size: 24px; font-weight: 700; color: #333; text-decoration: none;">%s</a>',
|
||||
esc_url($variables['store_url']),
|
||||
esc_html($header_text)
|
||||
);
|
||||
}
|
||||
|
||||
// Build full HTML
|
||||
$html = '<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>' . esc_html($subject) . '</title>
|
||||
<style>
|
||||
body { font-family: "Inter", Arial, sans-serif; background: #f8f8f8; margin: 0; padding: 20px; }
|
||||
.container { max-width: 600px; margin: 0 auto; }
|
||||
.header { padding: 32px; text-align: center; background: #f8f8f8; }
|
||||
.card-gutter { padding: 0 16px; }
|
||||
.card { background: #ffffff; border-radius: 8px; margin-bottom: 24px; padding: 32px 40px; width: 100%; box-sizing: border-box; }
|
||||
.card-hero { background: linear-gradient(135deg, ' . esc_attr($hero_gradient_start) . ' 0%, ' . esc_attr($hero_gradient_end) . ' 100%); color: #ffffff; }
|
||||
.card-hero * { color: #ffffff !important; }
|
||||
.card-success { background-color: #f0fdf4; }
|
||||
.card-info { background-color: #f0f7ff; }
|
||||
.card-warning { background-color: #fff8e1; }
|
||||
.card-basic { background: none; padding: 0; }
|
||||
h1, h2, h3 { margin-top: 0; color: #333; }
|
||||
p { font-size: 16px; line-height: 1.6; color: #555; margin-bottom: 16px; }
|
||||
.button { display: inline-block; background: ' . esc_attr($primary_color) . '; color: #ffffff !important; padding: 14px 28px; border-radius: 6px; text-decoration: none; font-weight: 600; }
|
||||
.button-outline { display: inline-block; background: transparent; color: ' . esc_attr($secondary_color) . ' !important; padding: 12px 26px; border: 2px solid ' . esc_attr($secondary_color) . '; border-radius: 6px; text-decoration: none; font-weight: 600; }
|
||||
.footer { padding: 32px; text-align: center; color: #888; font-size: 13px; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container">
|
||||
<div class="header">' . $header . '</div>
|
||||
<div class="card-gutter">' . $content . '</div>
|
||||
<div class="footer"><p>' . nl2br(esc_html($footer_text)) . '</p></div>
|
||||
</div>
|
||||
</body>
|
||||
</html>';
|
||||
|
||||
return $html;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse cards for test email
|
||||
*
|
||||
* @param string $content
|
||||
* @return string
|
||||
*/
|
||||
private function parse_cards_for_test($content) {
|
||||
// Parse [card:type] syntax
|
||||
$content = preg_replace_callback(
|
||||
'/\[card:(\w+)\](.*?)\[\/card\]/s',
|
||||
function($matches) {
|
||||
$type = $matches[1];
|
||||
$card_content = $this->markdown_to_html($matches[2]);
|
||||
return '<div class="card card-' . esc_attr($type) . '">' . $card_content . '</div>';
|
||||
},
|
||||
$content
|
||||
);
|
||||
|
||||
// Parse [card type="..."] syntax
|
||||
$content = preg_replace_callback(
|
||||
'/\[card([^\]]*)\](.*?)\[\/card\]/s',
|
||||
function($matches) {
|
||||
$attrs = $matches[1];
|
||||
$card_content = $this->markdown_to_html($matches[2]);
|
||||
$type = 'default';
|
||||
if (preg_match('/type=["\']([^"\']+)["\']/', $attrs, $type_match)) {
|
||||
$type = $type_match[1];
|
||||
}
|
||||
return '<div class="card card-' . esc_attr($type) . '">' . $card_content . '</div>';
|
||||
},
|
||||
$content
|
||||
);
|
||||
|
||||
// Parse buttons - new [button:style](url)Text[/button] syntax
|
||||
$content = preg_replace_callback(
|
||||
'/\[button:(\w+)\]\(([^)]+)\)([^\[]+)\[\/button\]/',
|
||||
function($matches) {
|
||||
$style = $matches[1];
|
||||
$url = $matches[2];
|
||||
$text = trim($matches[3]);
|
||||
$class = $style === 'outline' ? 'button-outline' : 'button';
|
||||
return '<p style="text-align: center;"><a href="' . esc_url($url) . '" class="' . $class . '">' . esc_html($text) . '</a></p>';
|
||||
},
|
||||
$content
|
||||
);
|
||||
|
||||
// Parse buttons - old [button url="..." style="..."]Text[/button] syntax
|
||||
$content = preg_replace_callback(
|
||||
'/\[button\s+url=["\']([^"\']+)["\'](?:\s+style=["\'](\w+)["\'])?\]([^\[]+)\[\/button\]/',
|
||||
function($matches) {
|
||||
$url = $matches[1];
|
||||
$style = $matches[2] ?? 'solid';
|
||||
$text = trim($matches[3]);
|
||||
$class = $style === 'outline' ? 'button-outline' : 'button';
|
||||
return '<p style="text-align: center;"><a href="' . esc_url($url) . '" class="' . $class . '">' . esc_html($text) . '</a></p>';
|
||||
},
|
||||
$content
|
||||
);
|
||||
|
||||
// If no cards found, wrap in default card
|
||||
if (strpos($content, '<div class="card') === false) {
|
||||
$content = '<div class="card">' . $this->markdown_to_html($content) . '</div>';
|
||||
}
|
||||
|
||||
return $content;
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic markdown to HTML conversion
|
||||
*
|
||||
* @param string $text
|
||||
* @return string
|
||||
*/
|
||||
private function markdown_to_html($text) {
|
||||
// Parse buttons FIRST - new [button:style](url)Text[/button] syntax
|
||||
$text = preg_replace_callback(
|
||||
'/\[button:(\w+)\]\(([^)]+)\)([^\[]+)\[\/button\]/',
|
||||
function($matches) {
|
||||
$style = $matches[1];
|
||||
$url = $matches[2];
|
||||
$btn_text = trim($matches[3]);
|
||||
$class = $style === 'outline' ? 'button-outline' : 'button';
|
||||
return '<p style="text-align: center;"><a href="' . esc_url($url) . '" class="' . $class . '">' . esc_html($btn_text) . '</a></p>';
|
||||
},
|
||||
$text
|
||||
);
|
||||
|
||||
// Parse buttons - old [button url="..."] syntax
|
||||
$text = preg_replace_callback(
|
||||
'/\[button\s+url=["\']([^"\']+)["\'](?:\s+style=[\'"](\\w+)[\'"])?\]([^\[]+)\[\/button\]/',
|
||||
function($matches) {
|
||||
$url = $matches[1];
|
||||
$style = $matches[2] ?? 'solid';
|
||||
$btn_text = trim($matches[3]);
|
||||
$class = $style === 'outline' ? 'button-outline' : 'button';
|
||||
return '<p style="text-align: center;"><a href="' . esc_url($url) . '" class="' . $class . '">' . esc_html($btn_text) . '</a></p>';
|
||||
},
|
||||
$text
|
||||
);
|
||||
|
||||
// Headers
|
||||
$text = preg_replace('/^### (.+)$/m', '<h3>$1</h3>', $text);
|
||||
$text = preg_replace('/^## (.+)$/m', '<h2>$1</h2>', $text);
|
||||
$text = preg_replace('/^# (.+)$/m', '<h1>$1</h1>', $text);
|
||||
|
||||
// Bold
|
||||
$text = preg_replace('/\*\*(.+?)\*\*/', '<strong>$1</strong>', $text);
|
||||
|
||||
// Italic
|
||||
$text = preg_replace('/\*(.+?)\*/', '<em>$1</em>', $text);
|
||||
|
||||
// Links (but not button syntax - already handled above)
|
||||
$text = preg_replace('/\[(?!button)([^\]]+)\]\(([^)]+)\)/', '<a href="$2">$1</a>', $text);
|
||||
|
||||
// List items
|
||||
$text = preg_replace('/^- (.+)$/m', '<li>$1</li>', $text);
|
||||
$text = preg_replace('/(<li>.*<\/li>)/s', '<ul>$1</ul>', $text);
|
||||
|
||||
// Paragraphs - wrap lines that aren't already wrapped
|
||||
$lines = explode("\n", $text);
|
||||
$result = [];
|
||||
foreach ($lines as $line) {
|
||||
$line = trim($line);
|
||||
if (empty($line)) continue;
|
||||
if (!preg_match('/^<(h[1-6]|ul|li|div|p|table|tr|td|th)/', $line)) {
|
||||
$line = '<p>' . $line . '</p>';
|
||||
}
|
||||
$result[] = $line;
|
||||
}
|
||||
|
||||
return implode("\n", $result);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user