/
var
/
www
/
barefootlaw.org
/
wp-content
/
plugins
/
newsletter
/
emails
/
Upload File
HOME
<?php defined('ABSPATH') || exit; class NewsletterEmails extends NewsletterModule { static $instance; const EDITOR_COMPOSER = 2; const EDITOR_HTML = 1; const EDITOR_TINYMCE = 0; static $PRESETS_LIST; const PRESET_EMAIL_TYPE = 'composer_template'; // Cache var $blocks = null; /** * @return NewsletterEmails */ static function instance() { if (self::$instance == null) { self::$instance = new NewsletterEmails(); } return self::$instance; } function __construct() { self::$PRESETS_LIST = array("cta", "invite", "announcement", "posts", "sales", "product", "tour", "simple"); $this->themes = new NewsletterThemes('emails'); parent::__construct('emails', '1.1.5'); add_action('newsletter_action', array($this, 'hook_newsletter_action'), 13, 3); add_action('newsletter_init', [$this, 'hook_newsletter_init']); if (is_admin()) { // Thank you to plugins which add the WP editor on other admin plugin pages... if (isset($_GET['page']) && $_GET['page'] == 'newsletter_emails_edit') { global $wp_actions; $wp_actions['wp_enqueue_editor'] = 1; } } } function hook_newsletter_init() { if (is_admin()) { if (defined('DOING_AJAX') && DOING_AJAX) { if (Newsletter::instance()->is_allowed()) { add_action('wp_ajax_tnpc_render', array($this, 'tnpc_render_callback')); add_action('wp_ajax_tnpc_preview', array($this, 'tnpc_preview_callback')); add_action('wp_ajax_tnpc_css', array($this, 'tnpc_css_callback')); add_action('wp_ajax_tnpc_options', array($this, 'hook_wp_ajax_tnpc_options')); add_action('wp_ajax_tnpc_get_all_presets', array($this, 'ajax_get_all_presets')); add_action('wp_ajax_tnpc_get_preset', array($this, 'ajax_get_preset')); add_action('wp_ajax_tnpc_delete_preset', array($this, 'hook_wp_ajax_tnpc_delete_preset')); add_action('wp_ajax_tnpc_regenerate_email', array($this, 'hook_wp_ajax_tnpc_regenerate_email')); } } } } function options_decode($options) { // Old "query string" format if (is_string($options) && strpos($options, 'options[') !== false) { $opts = []; parse_str($options, $opts); $options = $opts['options']; } if (is_array($options)) { return $options; } // Json data should be base64 encoded, but for short time it wasn't $tmp = json_decode($options, true); if (is_null($tmp)) { return json_decode(base64_decode($options), true); } else { return $tmp; } } /** * * @param array $options Options array */ function options_encode($options) { return base64_encode(json_encode($options, JSON_HEX_TAG | JSON_HEX_AMP)); } /** * Builds and returns the HTML with the form fields of a specific block. * * @global wpdb $wpdb */ function hook_wp_ajax_tnpc_options() { global $wpdb; $block = $this->get_block($_REQUEST['id']); if (!$block) { die('Block not found with id ' . esc_html($_REQUEST['id'])); } if (!class_exists('NewsletterControls')) { include NEWSLETTER_INCLUDES_DIR . '/controls.php'; } $options = $this->options_decode(stripslashes_deep($_REQUEST['options'])); $composer = isset($_POST['composer']) ? $_POST['composer'] : []; $context = array('type' => ''); if (isset($_REQUEST['context_type'])) { $context['type'] = $_REQUEST['context_type']; } $controls = new NewsletterControls($options); $fields = new NewsletterFields($controls); $controls->init(); echo '<input type="hidden" name="action" value="tnpc_render">'; echo '<input type="hidden" name="id" value="' . esc_attr($_REQUEST['id']) . '">'; echo '<input type="hidden" name="context_type" value="' . esc_attr($context['type']) . '">'; $inline_edits = ''; if (isset($controls->data['inline_edits'])) { $inline_edits = $controls->data['inline_edits']; } echo '<input type="hidden" name="options[inline_edits]" value="', esc_attr($this->options_encode($inline_edits)), '">'; echo "<h2>", esc_html($block["name"]), "</h2>"; include $block['dir'] . '/options.php'; wp_die(); } /** * Retrieves the presets list (no id in GET) or a specific preset id in GET) */ public function ajax_get_all_presets() { wp_send_json_success($this->get_all_preset()); } public function ajax_get_preset() { if (empty($_REQUEST['id'])) { wp_send_json_error([ 'msg' => __('Invalid preset ID') ]); } $preset_id = $_REQUEST['id']; $preset_content = $this->get_preset_content($preset_id); $global_options = $this->get_preset_global_options($preset_id); wp_send_json_success([ 'content' => $preset_content, 'globalOptions' => $global_options, ]); } private function get_preset_content($preset_id) { $content = ''; if ($this->is_a_tnp_default_preset($preset_id)) { // Get preset from file $preset = $this->get_preset_from_file($preset_id); foreach ($preset->blocks as $item) { ob_start(); $this->render_block($item->block, true, (array) $item->options); $content .= trim(ob_get_clean()); } } else { // Get preset from db $preset_email = $this->get_email(intval($preset_id)); $global_options = $this->extract_global_options_from($preset_email); $content = $this->regenerate_email_blocks($preset_email->message, $global_options); } return $content; } private function get_preset_global_options($preset_id) { if ($this->is_a_tnp_default_preset($preset_id)) { return []; } // Get preset from db $preset_email = $this->get_email(intval($preset_id)); $global_options = $this->extract_global_options_from($preset_email); return $global_options; } private function extract_global_options_from($email) { $global_options = []; foreach ($email->options as $global_option_name => $global_option) { if (strpos($global_option_name, 'composer_') === 0) { $global_options[str_replace('composer_', '', $global_option_name)] = $global_option; } } return $global_options; } private function is_a_tnp_default_preset($preset_id) { return in_array($preset_id, self::$PRESETS_LIST); } private function get_all_preset() { $content = "<div class='tnpc-preset-container'>"; if ($this->is_normal_context_request()) { $content .= "<div class='tnpc-preset-legacy-themes'><a href='" . $this->get_admin_page_url('theme') . "'>" . __('Looking for legacy themes?', 'newsletter') . "</a></div>"; } // LOAD USER PRESETS $user_preset_list = $this->get_emails(self::PRESET_EMAIL_TYPE); foreach ($user_preset_list as $user_preset) { $default_icon_url = NEWSLETTER_URL . "/emails/presets/default-icon.png?ver=2"; $preset_name = $user_preset->subject; $delete_preset_text = __('Delete', 'newsletter'); $edit_preset_text = __('Edit', 'newsletter'); // esc_js() assumes the string will be in single quote (arghhh!!!) $onclick_edit = 'tnpc_edit_preset(' . ((int) $user_preset->id) . ', \'' . esc_js($preset_name) . '\', event)'; $onclick_delete = 'tnpc_delete_preset(' . ((int) $user_preset->id) . ', \'' . esc_js($preset_name) . '\', event)'; $onclick_load = 'tnpc_load_preset(' . ((int) $user_preset->id) . ', \'' . esc_js($preset_name) . '\', event)'; $content .= "<div class='tnpc-preset' onclick='" . esc_attr($onclick_load) . "'>\n"; $content .= "<img src='$default_icon_url' title='" . esc_attr($preset_name) . "' alt='" . esc_attr($preset_name) . "'>\n"; $content .= "<span class='tnpc-preset-label'>" . esc_html($user_preset->subject) . "</span>\n"; $content .= "<span class='tnpc-delete-preset' onclick='" . esc_attr($onclick_delete) . "' title='" . esc_attr($delete_preset_text) . "'><i class='fas fa-times'></i></span>\n"; $content .= "<span class='tnpc-edit-preset' onclick='" . esc_attr($onclick_edit) . "' title='" . esc_attr($edit_preset_text) . "'><i class='fas fa-pencil-alt'></i></span>\n"; $content .= "</div>"; } // LOAD TNP PRESETS foreach (self::$PRESETS_LIST as $id) { $preset = $this->get_preset_from_file($id); $preset_name = esc_html($preset->name); $content .= "<div class='tnpc-preset' onclick='tnpc_load_preset(\"$id\")'>"; $content .= "<img src='$preset->icon' title='$preset_name' alt='$preset_name'/>"; $content .= "<span class='tnpc-preset-label'>$preset_name</span>"; $content .= "</div>"; } if ($this->is_normal_context_request()) { $content .= $this->get_automated_spot_element(); $content .= $this->get_autoresponder_spot_element(); $content .= $this->get_raw_html_preset_element(); } return $content; } private function is_normal_context_request() { return empty($_REQUEST['context_type']); } private function is_automated_context_request() { return isset($_REQUEST['context_type']) && $_REQUEST['context_type'] === 'automated'; } private function is_autoresponder_context_request() { return isset($_REQUEST['context_type']) && $_REQUEST['context_type'] === 'autoresponder'; } private function get_automated_spot_element() { $result = "<div class='tnpc-preset'>"; if (class_exists('NewsletterAutomated')) { $result .= "<a href='?page=newsletter_automated_index'>"; } else { $result .= "<a href='https://www.thenewsletterplugin.com/automated?utm_source=composer&utm_campaign=plugin&utm_medium=automated'>"; } $result .= "<img src='" . plugins_url('newsletter') . "/emails/images/automated.png' title='Automated addon' alt='Automated'/>"; $result .= "<span class='tnpc-preset-label'>Daily, weekly and monthly newsletters</span></a>"; $result .= "</div>"; return $result; } private function get_autoresponder_spot_element() { $result = "<div class='tnpc-preset'>"; if (class_exists('NewsletterAutoresponder')) { $result .= "<a href='?page=newsletter_autoresponder_index'>"; } else { $result .= "<a href='https://www.thenewsletterplugin.com/autoresponder?utm_source=composer&utm_campaign=plugin&utm_medium=autoresponder' target='_blank'>"; } $result .= "<img src='" . plugins_url('newsletter') . "/emails/images/autoresponder.png' title='Autoresponder addon' alt='Autoresponder'/>"; $result .= "<span class='tnpc-preset-label'>Autoresponders</span></a>"; $result .= "</div>"; return $result; } private function get_raw_html_preset_element() { $result = "<div class='tnpc-preset tnpc-preset-html' onclick='location.href=\"" . wp_nonce_url('admin.php?page=newsletter_emails_new&id=rawhtml', 'newsletter-new') . "\"'>"; $result .= "<img src='" . plugins_url('newsletter') . "/emails/images/rawhtml.png' title='RAW HTML' alt='RAW'/>"; $result .= "<span class='tnpc-preset-label'>Raw HTML</span>"; $result .= "</div>"; $result .= "<div class='clear'></div>"; $result .= "</div>"; return $result; } function has_dynamic_blocks($theme) { preg_match_all('/data-json="(.*?)"/m', $theme, $matches, PREG_PATTERN_ORDER); foreach ($matches[1] as $match) { $a = html_entity_decode($match, ENT_QUOTES, 'UTF-8'); $options = $this->options_decode($a); $block = $this->get_block($options['block_id']); if (!$block) { continue; } if ($block['type'] == 'dynamic') { return true; } } return false; } /** * Regenerates a saved composed email rendering each block. Regeneration is * conditioned (possibly) by the context. The context is usually passed to blocks * so they can act in the right manner. * * $context contains a type and, for automated, the last_run. * * $email can actually be even a string containing the full newsletter HTML code. * * @param TNP_Email $email * @return string */ function regenerate($email, $context = []) { $context = array_merge(['last_run' => 0, 'type' => ''], $context); $composer = []; foreach ($email->options as $k=>$v) { if (strpos($k, 'composer_') !== 0) continue; $composer[substr($k, 9)] = $v; } preg_match_all('/data-json="(.*?)"/m', $email->message, $matches, PREG_PATTERN_ORDER); $result = ''; $subject = ''; foreach ($matches[1] as $match) { $a = html_entity_decode($match, ENT_QUOTES, 'UTF-8'); $options = $this->options_decode($a); $block = $this->get_block($options['block_id']); if (!$block) { $this->logger->debug('Unable to load the block ' . $options['block_id']); //continue; } ob_start(); $out = $this->render_block($options['block_id'], true, $options, $context, $composer); if (is_array($out)) { if ($out['return_empty_message'] || $out['stop']) { return false; } if ($out['skip']) { continue; } if (empty($subject) && !empty($out['subject'])) { $subject = strip_tags($out['subject']); } } $block_html = ob_get_clean(); $result .= $block_html; } $email->message = TNP_Composer::get_html_open($email) . TNP_Composer::get_main_wrapper_open($email) . $result . TNP_Composer::get_main_wrapper_close($email) . TNP_Composer::get_html_close($email); $email->subject = $subject; return true; } function remove_block_data($text) { // TODO: Lavorare! return $text; } static function get_outlook_wrapper_open($width = 600) { return '<!--[if mso | IE]><table role="presentation" border="0" cellpadding="0" align="center" cellspacing="0" width="' . $width . '"><tr><td width="' . $width . '" style="vertical-align:top;width:' . $width . 'px;"><![endif]-->'; } static function get_outlook_wrapper_close() { return "<!--[if mso | IE]></td></tr></table><![endif]-->"; } function hook_safe_style_css($rules) { $rules[] = 'display'; return $rules; } /** * Renders a block identified by its id, using the block options and adding a wrapper * if required (for the first block rendering). * * @param string $block_id * @param boolean $wrapper * @param array $options * @param array $context * @param array $composer */ function render_block($block_id = null, $wrapper = false, $options = [], $context = [], $composer = []) { static $kses_style_filter = false; include_once NEWSLETTER_INCLUDES_DIR . '/helper.php'; //Remove 'options_composer_' prefix $composer_defaults = []; foreach (TNP_Composer::get_global_style_defaults() as $global_option_name => $global_option) { $composer_defaults[str_replace('options_composer_', '', $global_option_name)] = $global_option; } $composer = array_merge($composer_defaults, $composer); $width = 600; $font_family = 'Helvetica, Arial, sans-serif'; $global_title_font_family = $composer['title_font_family']; $global_title_font_size = $composer['title_font_size']; $global_title_font_color = $composer['title_font_color']; $global_title_font_weight = $composer['title_font_weight']; $global_text_font_family = $composer['text_font_family']; $global_text_font_size = $composer['text_font_size']; $global_text_font_color = $composer['text_font_color']; $global_text_font_weight = $composer['text_font_weight']; $global_button_font_family = $composer['button_font_family']; $global_button_font_size = $composer['button_font_size']; $global_button_font_color = $composer['button_font_color']; $global_button_font_weight = $composer['button_font_weight']; $global_button_background_color = $composer['button_background_color']; $global_block_background = $composer['block_background']; $info = Newsletter::instance()->get_options('info'); // Just in case... if (!is_array($options)) { $options = array(); } // This code filters the HTML to remove javascript and unsecure attributes and enable the // "display" rule for CSS which is needed in blocks to force specific "block" or "inline" or "table". add_filter('safe_style_css', [$this, 'hook_safe_style_css'], 9999); $options = wp_kses_post_deep($options); remove_filter('safe_style_css', [$this, 'hook_safe_style_css']); $block_options = get_option('newsletter_main'); $block = $this->get_block($block_id); if (!isset($context['type'])) $context['type'] = ''; // Block not found if (!$block) { if ($wrapper) { echo '<table border="0" cellpadding="0" cellspacing="0" align="center" width="100%" style="border-collapse: collapse; width: 100%;" class="tnpc-row tnpc-row-block" data-id="', esc_attr($block_id), '">'; echo '<tr>'; echo '<td data-options="" bgcolor="#ffffff" align="center" style="padding: 0; font-family: Helvetica, Arial, sans-serif;" class="edit-block">'; } echo $this->get_outlook_wrapper_open($width); echo '<p>Ops, this block type is not avalable.</p>'; echo $this->get_outlook_wrapper_close(); if ($wrapper) { echo '</td></tr></table>'; } return; } $out = ['subject' => '', 'return_empty_message' => false, 'stop' => false, 'skip' => false]; $dir = is_rtl() ? 'rtl' : 'ltr'; $align_left = is_rtl() ? 'right' : 'left'; $align_right = is_rtl() ? 'left' : 'right'; ob_start(); $logger = $this->logger; include $block['dir'] . '/block.php'; $content = trim(ob_get_clean()); if (empty($content)) { return $out; } $common_defaults = array( 'block_padding_top' => 0, 'block_padding_bottom' => 0, 'block_padding_right' => 0, 'block_padding_left' => 0, 'block_background' => '', 'block_background_2' => '', 'block_width' => 600, 'block_align' => 'center' ); $options = array_merge($common_defaults, $options); // Obsolete $content = str_replace('{width}', $width, $content); $content = $this->inline_css($content, true); // CSS driven by the block // Requited for the server side parsing and rendering $options['block_id'] = $block_id; $options['block_padding_top'] = (int) str_replace('px', '', $options['block_padding_top']); $options['block_padding_bottom'] = (int) str_replace('px', '', $options['block_padding_bottom']); $options['block_padding_right'] = (int) str_replace('px', '', $options['block_padding_right']); $options['block_padding_left'] = (int) str_replace('px', '', $options['block_padding_left']); $block_background = empty($options['block_background']) ? $global_block_background : $options['block_background']; // Internal TD wrapper $style = 'text-align: center; '; $style .= 'width: 100% !important; '; $style .= 'line-height: normal !important; '; $style .= 'letter-spacing: normal; '; $style .= 'padding-top: ' . $options['block_padding_top'] . 'px; '; $style .= 'padding-left: ' . $options['block_padding_left'] . 'px; '; $style .= 'padding-right: ' . $options['block_padding_right'] . 'px; '; $style .= 'padding-bottom: ' . $options['block_padding_bottom'] . 'px; '; $style .= 'background-color: ' . $block_background . ';'; if (isset($options['block_background_gradient'])) { $style .= 'background: linear-gradient(180deg, ' . $block_background . ' 0%, ' . $options['block_background_2'] . ' 100%);'; } $data = $this->options_encode($options); // First time block creation wrapper if ($wrapper) { echo '<table border="0" cellpadding="0" cellspacing="0" align="center" width="100%" style="border-collapse: collapse; width: 100%;" class="tnpc-row tnpc-row-block" data-id="', esc_attr($block_id), '">', "\n"; echo "<tr>"; echo '<td align="center" style="padding: 0;" class="edit-block">', "\n"; } // Container that fixes the width and makes the block responsive echo $this->get_outlook_wrapper_open($options['block_width']); echo '<table type="options" data-json="', esc_attr($data), '" class="tnpc-block-content" border="0" cellpadding="0" align="center" cellspacing="0" width="100%" style="width: 100%!important; max-width: ', $options['block_width'], 'px!important">', "\n"; echo "<tr>"; echo '<td align="', esc_attr($options['block_align']), '" style="', esc_attr($style), '" bgcolor="', esc_attr($block_background), '" width="100%">'; //echo "<!-- block generated content -->\n"; echo trim($content); //echo "\n<!-- /block generated content -->\n"; echo "</td></tr></table>"; echo $this->get_outlook_wrapper_close(); // First time block creation wrapper if ($wrapper) { echo "</td></tr></table>"; } return $out; } /** * Ajax call to render a block with a new set of options after the settings popup * has been saved. * * @param type $block_id * @param type $wrapper */ function tnpc_render_callback() { if (!check_ajax_referer('save')) { $this->dienow('Expired request'); } $block_id = $_POST['id']; $wrapper = isset($_POST['full']); $options = $this->restore_options_from_request(); $this->render_block($block_id, $wrapper, $options, [], $_POST['composer']); wp_die(); } function hook_wp_ajax_tnpc_regenerate_email() { $content = stripslashes($_POST['content']); $global_options = $_POST['composer']; $regenerated_content = $this->regenerate_email_blocks($content, $global_options); wp_send_json_success([ 'content' => $regenerated_content, 'message' => __('Successfully updated', 'newsletter') ]); } private function regenerate_email_blocks($content, $global_options) { $raw_block_options = $this->extract_encoded_blocks_options($content); $regenerated_content = ''; foreach ($raw_block_options as $raw_block_option) { /* $a = html_entity_decode( $raw_block_option, ENT_QUOTES, 'UTF-8' ); $block_options = $this->options_decode( $a ); */ $block_options = $this->options_decode($raw_block_option); $block = $this->get_block($block_options['block_id']); if (!$block) { $this->logger->debug('Unable to load the block ' . $block_options['block_id']); } ob_start(); $this->render_block($block_options['block_id'], true, $block_options, [], $global_options); $block_html = ob_get_clean(); $regenerated_content .= $block_html; } return $regenerated_content; } /** * @param string $html_email_content Email html content * * @return string[] Encoded options of email blocks */ private function extract_encoded_blocks_options($html_email_content) { preg_match_all('/data-json="(.*?)"/m', $html_email_content, $raw_block_options, PREG_PATTERN_ORDER); return $raw_block_options[1]; } function tnpc_preview_callback() { $email = Newsletter::instance()->get_email($_REQUEST['id'], ARRAY_A); if (empty($email)) { echo 'Wrong email identifier'; return; } echo $email['message']; wp_die(); } function tnpc_css_callback() { include NEWSLETTER_DIR . '/emails/tnp-composer/css/newsletter.css'; wp_die(); } /** Returns the correct admin page to edit the newsletter with the correct editor. */ function get_editor_url($email_id, $editor_type) { switch ($editor_type) { case NewsletterEmails::EDITOR_COMPOSER: return admin_url("admin.php") . '?page=newsletter_emails_composer&id=' . $email_id; case NewsletterEmails::EDITOR_HTML: return admin_url("admin.php") . '?page=newsletter_emails_editorhtml&id=' . $email_id; case NewsletterEmails::EDITOR_TINYMCE: return admin_url("admin.php") . '?page=newsletter_emails_editortinymce&id=' . $email_id; } } /** * Returns the button linked to the correct "edit" page for the passed newsletter. The edit page can be an editor * or the targeting page (it depends on newsletter status). * * @param TNP_Email $email */ function get_edit_button($email, $only_icon = false) { $editor_type = $this->get_editor_type($email); if ($email->status == 'new') { $edit_url = $this->get_editor_url($email->id, $editor_type); } else { $edit_url = 'admin.php?page=newsletter_emails_edit&id=' . $email->id; } switch ($editor_type) { case NewsletterEmails::EDITOR_COMPOSER: $icon_class = 'th-large'; break; case NewsletterEmails::EDITOR_HTML: $icon_class = 'code'; break; default: $icon_class = 'edit'; break; } if ($only_icon) { return '<a class="button-primary" href="' . $edit_url . '" title="' . esc_attr__('Edit', 'newsletter') . '">' . '<i class="fas fa-' . $icon_class . '"></i></a>'; } else { return '<a class="button-primary" href="' . $edit_url . '" title="' . esc_attr__('Edit', 'newsletter') . '">' . '<i class="fas fa-' . $icon_class . '"></i> ' . __('Edit', 'newsletter') . '</a>'; } } /** Returns the correct editor type for the provided newsletter. Contains backward compatibility code. */ function get_editor_type($email) { $email = (object) $email; $editor_type = $email->editor; // Backward compatibility $email_options = maybe_unserialize($email->options); if (isset($email_options['composer'])) { $editor_type = NewsletterEmails::EDITOR_COMPOSER; } // End backward compatibility return $editor_type; } /** * * @param type $action * @param type $user * @param type $email * @return type * @global wpdb $wpdb */ function hook_newsletter_action($action, $user, $email) { global $wpdb; switch ($action) { case 'v': case 'view': $id = $_GET['id']; if ($id == 'last') { $email = $wpdb->get_row("select * from " . NEWSLETTER_EMAILS_TABLE . " where private=0 and type='message' and status='sent' order by send_on desc limit 1"); } else { $email = $this->get_email($_GET['id']); } if (empty($email)) { header("HTTP/1.0 404 Not Found"); die('Email not found'); } if (!Newsletter::instance()->is_allowed()) { if ($email->status == 'new') { header("HTTP/1.0 404 Not Found"); die('Not sent yet'); } if ($email->private == 1) { if (!$user) { header("HTTP/1.0 404 Not Found"); die('No available for online view'); } $sent = $wpdb->get_row($wpdb->prepare("select * from " . NEWSLETTER_SENT_TABLE . " where email_id=%d and user_id=%d limit 1", $email->id, $user->id)); if (!$sent) { header("HTTP/1.0 404 Not Found"); die('No available for online view'); } } } header('Content-Type: text/html;charset=UTF-8'); header('X-Robots-Tag: noindex,nofollow,noarchive'); header('Cache-Control: no-cache,no-store,private'); echo apply_filters('newsletter_view_message', $this->replace($email->message, $user, $email)); die(); break; case 'emails-css': $email_id = (int) $_GET['id']; $body = Newsletter::instance()->get_email_field($email_id, 'message'); $x = strpos($body, '<style'); if ($x === false) return; $x = strpos($body, '>', $x); $y = strpos($body, '</style>'); header('Content-Type: text/css;charset=UTF-8'); echo substr($body, $x + 1, $y - $x - 1); die(); break; case 'emails-composer-css': header('Cache: no-cache'); header('Content-Type: text/css'); echo $this->get_composer_css(); die(); break; case 'emails-preview': if (!Newsletter::instance()->is_allowed()) { die('Not enough privileges'); } if (!check_admin_referer('view')) { die(); } $theme_id = $_GET['id']; $theme = $this->themes->get_theme($theme_id); // Used by theme code $theme_options = $this->themes->get_options($theme_id); $theme_url = $theme['url']; header('Content-Type: text/html;charset=UTF-8'); include $theme['dir'] . '/theme.php'; die(); break; case 'emails-preview-text': header('Content-Type: text/plain;charset=UTF-8'); if (!Newsletter::instance()->is_allowed()) { die('Not enough privileges'); } if (!check_admin_referer('view')) { die(); } // Used by theme code $theme_options = $this->get_current_theme_options(); $file = include $theme['dir'] . '/theme-text.php'; if (is_file($file)) { include $file; } die(); break; case 'emails-create': // Newsletter from themes are created on frontend context because sometime WP themes change the way the content, // excerpt, thumbnail are extracted. if (!Newsletter::instance()->is_allowed()) { die('Not enough privileges'); } require_once NEWSLETTER_INCLUDES_DIR . '/controls.php'; $controls = new NewsletterControls(); if (!$controls->is_action('create')) { die('Wrong call'); } $theme_id = $controls->data['id']; $theme = $this->themes->get_theme($theme_id); if (!$theme) { die('invalid theme'); } $this->themes->save_options($theme_id, $controls->data); $email = array(); $email['status'] = 'new'; $email['subject'] = ''; //__('Here the email subject', 'newsletter'); $email['track'] = Newsletter::instance()->options['track']; $email['send_on'] = time(); $email['editor'] = NewsletterEmails::EDITOR_TINYMCE; $email['type'] = 'message'; $theme_options = $this->themes->get_options($theme_id); $theme_url = $theme['url']; $theme_subject = ''; ob_start(); include $theme['dir'] . '/theme.php'; $email['message'] = ob_get_clean(); if (!empty($theme_subject)) { $email['subject'] = $theme_subject; } if (file_exists($theme['dir'] . '/theme-text.php')) { ob_start(); include $theme['dir'] . '/theme-text.php'; $email['message_text'] = ob_get_clean(); } else { $email['message_text'] = 'You need a modern email client to read this email. Read it online: {email_url}.'; } $email = $this->save_email($email); $edit_url = $this->get_editor_url($email->id, $email->editor); header('Location: ' . $edit_url); die(); break; } } function admin_menu() { $this->add_menu_page('index', 'Newsletters'); $this->add_admin_page('list', 'Email List'); $this->add_admin_page('new', 'Email New'); $this->add_admin_page('edit', 'Email Edit'); $this->add_admin_page('theme', 'Email Themes'); $this->add_admin_page('composer', 'The Composer'); $this->add_admin_page('editorhtml', 'HTML Editor'); $this->add_admin_page('editortinymce', 'TinyMCE Editor'); } /** * Builds a block data structure starting from the folder containing the block * files. * * @param string $dir * @return array | WP_Error */ function build_block($dir) { $dir = realpath($dir); $dir = wp_normalize_path($dir); $full_file = $dir . '/block.php'; if (!is_file($full_file)) { return new WP_Error('1', 'Missing block.php file in ' . $dir); } $wp_content_dir = wp_normalize_path(realpath(WP_CONTENT_DIR)); $relative_dir = substr($dir, strlen($wp_content_dir)); $file = basename($dir); $data = get_file_data($full_file, ['name' => 'Name', 'section' => 'Section', 'description' => 'Description', 'type' => 'Type']); $defaults = ['section' => 'content', 'name' => ucfirst($file), 'descritpion' => '', 'icon' => plugins_url('newsletter') . '/admin/images/block-icon.png']; $data = array_merge($defaults, $data); if (is_file($dir . '/icon.png')) { $data['icon'] = content_url($relative_dir . '/icon.png'); } $data['id'] = sanitize_key($file); // Absolute path of the block files $data['dir'] = $dir; $data['url'] = content_url($relative_dir); return $data; } /** * * @param type $dir * @return type */ function scan_blocks_dir($dir) { $dir = realpath($dir); if (!$dir) { return []; } $dir = wp_normalize_path($dir); $list = []; $handle = opendir($dir); while ($file = readdir($handle)) { $data = $this->build_block($dir . '/' . $file); if (is_wp_error($data)) { $this->logger->error($data); continue; } $list[$data['id']] = $data; } closedir($handle); return $list; } /** * Array of arrays with every registered block and legacy block converted to the new * format. * * @return array */ function get_blocks() { if (!is_null($this->blocks)) { return $this->blocks; } $this->blocks = $this->scan_blocks_dir(__DIR__ . '/blocks'); $extended = $this->scan_blocks_dir(WP_CONTENT_DIR . '/extensions/newsletter/blocks'); $this->blocks = array_merge($extended, $this->blocks); $dirs = apply_filters('newsletter_blocks_dir', array()); //$this->logger->debug('Block dirs:'); //$this->logger->debug($dirs); foreach ($dirs as $dir) { $list = $this->scan_blocks_dir($dir); $this->blocks = array_merge($list, $this->blocks); } do_action('newsletter_register_blocks'); foreach (TNP_Composer::$block_dirs as $dir) { $block = $this->build_block($dir); if (is_wp_error($block)) { $this->logger->error($block); continue; } if (!isset($this->blocks[$block['id']])) { $this->blocks[$block['id']] = $block; } else { $this->logger->error('The block "' . $block['id'] . '" has already been registered'); } } $this->blocks = array_reverse($this->blocks); return $this->blocks; } /** * Return a single block (associative array) checking for legacy ID as well. * * @param string $id * @return array */ function get_block($id) { switch ($id) { case 'content-03-text.block': $id = 'text'; break; case 'footer-03-social.block': $id = 'social'; break; case 'footer-02-canspam.block': $id = 'canspam'; break; case 'content-05-image.block': $id = 'image'; break; case 'header-01-header.block': $id = 'header'; break; case 'footer-01-footer.block': $id = 'footer'; break; case 'content-02-heading.block': $id = 'heading'; break; case 'content-07-twocols.block': case 'content-06-posts.block': $id = 'posts'; break; case 'content-04-cta.block': $id = 'cta'; break; case 'content-01-hero.block': $id = 'hero'; break; // case 'content-02-heading.block': $id = '/plugins/newsletter/emails/blocks/heading'; // break; } // Conversion for old full path ID $id = sanitize_key(basename($id)); // TODO: Correct id for compatibility $blocks = $this->get_blocks(); if (!isset($blocks[$id])) { return null; } return $blocks[$id]; } function scan_presets_dir($dir = null) { if (is_null($dir)) { $dir = __DIR__ . '/presets'; } if (!is_dir($dir)) { return array(); } $handle = opendir($dir); $list = array(); $relative_dir = substr($dir, strlen(WP_CONTENT_DIR)); while ($file = readdir($handle)) { if ($file == '.' || $file == '..') continue; // The block unique key, we should find out how to build it, maybe an hash of the (relative) dir? $preset_id = sanitize_key($file); $full_file = $dir . '/' . $file . '/preset.json'; if (!is_file($full_file)) { continue; } $icon = content_url($relative_dir . '/' . $file . '/icon.png'); $list[$preset_id] = $icon; } closedir($handle); return $list; } function get_preset_from_file($id, $dir = null) { if (is_null($dir)) { $dir = __DIR__ . '/presets'; } $id = $this->sanitize_file_name($id); if (!is_dir($dir . '/' . $id) || !in_array($id, self::$PRESETS_LIST)) { return array(); } $json_content = file_get_contents("$dir/$id/preset.json"); $json_content = str_replace("{placeholder_base_url}", plugins_url('newsletter') . '/emails/presets', $json_content); $json = json_decode($json_content); $json->icon = NEWSLETTER_URL . "/emails/presets/$id/icon.png?ver=2"; return $json; } function get_composer_css() { $css = file_get_contents(__DIR__ . '/tnp-composer/css/newsletter.css'); $blocks = $this->get_blocks(); foreach ($blocks as $block) { if (!file_exists($block['dir'] . '/style.css')) { continue; } $css .= "\n\n"; $css .= "/* " . $block['name'] . " */\n"; $css .= file_get_contents($block['dir'] . '/style.css'); } return $css; } function get_composer_backend_css() { $css = file_get_contents(__DIR__ . '/tnp-composer/css/backend.css'); $css .= "\n\n"; $css .= $this->get_composer_css(); return $css; } /** * Send an email to the test subscribers. * * @param TNP_Email $email Could be any object with the TNP_Email attributes * @param NewsletterControls $controls */ function send_test_email($email, $controls) { if (!$email) { $controls->errors = __('Newsletter should be saved before send a test', 'newsletter'); return; } $original_subject = $email->subject; $this->set_test_subject_to($email); $users = NewsletterUsers::instance()->get_test_users(); if (count($users) == 0) { $controls->errors = '' . __('There are no test subscribers to send to', 'newsletter') . '. <a href="https://www.thenewsletterplugin.com/plugins/newsletter/subscribers-module#test" target="_blank"><strong>' . __('Read more', 'newsletter') . '</strong></a>.'; } else { $r = Newsletter::instance()->send($email, $users, true); $emails = array(); foreach ($users as $user) { $emails[] = '<a href="admin.php?page=newsletter_users_edit&id=' . $user->id . '" target="_blank">' . $user->email . '</a>'; } if (is_wp_error($r)) { $controls->errors = 'Something went wrong. Check the error logs on status page.<br>'; $controls->errors .= __('Test subscribers:', 'newsletter'); $controls->errors .= ' ' . implode(', ', $emails); $controls->errors .= '<br>'; $controls->errors .= '<strong>' . esc_html($r->get_error_message()) . '</strong><br>'; $controls->errors .= '<a href="https://www.thenewsletterplugin.com/documentation/email-sending-issues" target="_blank"><strong>' . __('Read more about delivery issues', 'newsletter') . '</strong></a>.'; } else { $controls->messages = __('Test subscribers:', 'newsletter'); $controls->messages .= ' ' . implode(', ', $emails); $controls->messages .= '.<br>'; $controls->messages .= '<a href="https://www.thenewsletterplugin.com/documentation/subscribers#test" target="_blank"><strong>' . __('Read more about test subscribers', 'newsletter') . '</strong></a>.<br>'; $controls->messages .= '<a href="https://www.thenewsletterplugin.com/documentation/email-sending-issues" target="_blank"><strong>' . __('Read more about delivery issues', 'newsletter') . '</strong></a>.'; } } $email->subject = $original_subject; } /** * Send an email to the test subscribers. * * @param TNP_Email $email Could be any object with the TNP_Email attributes * @param string $email_address * * @throws Exception */ function send_test_newsletter_to_email_address($email, $email_address) { if (!$email) { throw new Exception(__('Newsletter should be saved before send a test', 'newsletter')); } $this->set_test_subject_to($email); $dummy_subscriber = $this->make_dummy_subscriber(); $dummy_subscriber->email = $email_address; $result = Newsletter::instance()->send($email, [$dummy_subscriber], true); $email = '<a href="admin.php?page=newsletter_users_edit&id=' . $dummy_subscriber->id . '" target="_blank">' . $dummy_subscriber->email . '</a>'; if (is_wp_error($result)) { $error_message = 'Something went wrong. Check the error logs on status page.<br>'; $error_message .= __('Test subscribers:', 'newsletter'); $error_message .= ' ' . $email; $error_message .= '<br>'; $error_message .= '<strong>' . esc_html($result->get_error_message()) . '</strong><br>'; $error_message .= '<a href="https://www.thenewsletterplugin.com/documentation/email-sending-issues" target="_blank"><strong>' . __('Read more about delivery issues', 'newsletter') . '</strong></a>.'; throw new Exception($error_message); } $messages = __('Test subscribers:', 'newsletter'); $messages .= ' ' . $email; $messages .= '.<br>'; $messages .= '<a href="https://www.thenewsletterplugin.com/documentation/subscribers#test" target="_blank"><strong>' . __('Read more about test subscribers', 'newsletter') . '</strong></a>.<br>'; $messages .= '<a href="https://www.thenewsletterplugin.com/documentation/email-sending-issues" target="_blank"><strong>' . __('Read more about delivery issues', 'newsletter') . '</strong></a>.'; return $messages; } private function set_test_subject_to($email) { if ($email->subject == '') { $email->subject = '[TEST] Dummy subject, it was empty (remember to set it)'; } else { $email->subject = $email->subject . ' (TEST)'; } } private function make_dummy_subscriber() { $dummy_user = new TNP_User(); $dummy_user->id = 0; $dummy_user->email = 'john.doe@example.org'; $dummy_user->name = 'John'; $dummy_user->surname = 'Doe'; $dummy_user->sex = 'n'; $dummy_user->language = ''; $dummy_user->ip = ''; for ($i = 1; $i <= NEWSLETTER_PROFILE_MAX; $i++) { $profile_key = "profile_$i"; $dummy_user->$profile_key = ''; } return $dummy_user; } function restore_options_from_request() { require_once NEWSLETTER_INCLUDES_DIR . '/controls.php'; $controls = new NewsletterControls(); $options = $controls->data; if (isset($_POST['options']) && is_array($_POST['options'])) { // Get all block options //$options = stripslashes_deep($_POST['options']); // Deserialize inline edits when // render is preformed on saving block options if (isset($options['inline_edits']) && !is_array($options['inline_edits'])) { $options['inline_edits'] = $this->options_decode($options['inline_edits']); } // Restore inline edits from data-json // coming from inline editing // and merge with current inline edit if (isset($_POST['encoded_options'])) { $decoded_options = $this->options_decode($_POST['encoded_options']); $to_merge_inline_edits = []; if (isset($decoded_options['inline_edits'])) { foreach ($decoded_options['inline_edits'] as $decoded_inline_edit) { $to_merge_inline_edits[$decoded_inline_edit['post_id'] . $decoded_inline_edit['type']] = $decoded_inline_edit; } } //Overwrite with new edited content if (isset($options['inline_edits'])) { foreach ($options['inline_edits'] as $inline_edit) { $to_merge_inline_edits[$inline_edit['post_id'] . $inline_edit['type']] = $inline_edit; } } $options['inline_edits'] = array_values($to_merge_inline_edits); $options = array_merge($decoded_options, $options); } return $options; } return array(); } public function hook_wp_ajax_tnpc_delete_preset() { if (!wp_verify_nonce($_POST['_wpnonce'], 'preset')) { wp_send_json_error('Expired request'); } $preset_id = (int) $_REQUEST['presetId']; $newsletter = Newsletter::instance(); if ($preset_id > 0) { $preset = $newsletter->get_email($preset_id); if ($preset && $preset->type === self::PRESET_EMAIL_TYPE) { Newsletter::instance()->delete_email($preset_id); wp_send_json_success(); } else { wp_send_json_error(__('Is not a preset!', 'newsletter')); } } else { wp_send_json_error(); } } } NewsletterEmails::instance();