accessControl = $accessControl; $this->wp = $wp; $this->servicesChecker = $servicesChecker; $this->container = $container; $this->router = $router; $this->customFonts = $customFonts; } public function init() { $this->checkPremiumKey(); $this->wp->addAction( 'admin_menu', [ $this, 'setup', ] ); $this->wp->addFilter('parent_file', [$this, 'highlightNestedMailPoetSubmenus']); } public function setup() { global $parent_file; $parent_file = self::EMAILS_PAGE_SLUG; if (!$this->accessControl->validatePermission(AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN)) return; $this->router->checkRedirects(); $this->registerMailPoetMenu(); // @ToDo Remove Beta once Automation is no longer beta. $this->wp->addAction('admin_head', function () { echo ''; }); if (!self::isOnMailPoetAdminPage()) { return; } $this->wp->doAction('mailpoet_conflict_resolver_styles'); $this->wp->doAction('mailpoet_conflict_resolver_scripts'); if ( !isset($_REQUEST['page']) || sanitize_text_field(wp_unslash($_REQUEST['page'])) !== 'mailpoet-newsletter-editor' ) { return; } // Disable WP emojis to not interfere with the newsletter editor emoji handling $this->disableWPEmojis(); if (!$this->customFonts->displayCustomFonts()) { return; } $this->wp->addAction('admin_head', function () { echo ''; }); } private function registerMailPoetMenu() { // Main page $this->wp->addMenuPage( 'MailPoet', 'MailPoet', AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::MAIN_PAGE_SLUG, '', self::ICON_BASE64_SVG, 30 ); // Welcome wizard page $this->wp->addSubmenuPage( self::NO_PARENT_PAGE_SLUG, $this->setPageTitle(__('Welcome Wizard', 'mailpoet')), esc_html__('Welcome Wizard', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::WELCOME_WIZARD_PAGE_SLUG, [ $this, 'welcomeWizard', ] ); // Landingpage $this->wp->addSubmenuPage( self::NO_PARENT_PAGE_SLUG, $this->setPageTitle(__('MailPoet', 'mailpoet')), esc_html__('MailPoet', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::LANDINGPAGE_PAGE_SLUG, [ $this, 'landingPage', ] ); $this->registerMailPoetSubMenuEntries(); } private function registerMailPoetSubMenuEntries() { // Homepage $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Home', 'mailpoet')), esc_html__('Home', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::HOMEPAGE_PAGE_SLUG, [ $this, 'homepage', ] ); // Emails page $newslettersPage = $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Emails', 'mailpoet')), esc_html__('Emails', 'mailpoet'), AccessControl::PERMISSION_MANAGE_EMAILS, self::EMAILS_PAGE_SLUG, [ $this, 'newsletters', ] ); // add limit per page to screen options $this->wp->addAction('load-' . $newslettersPage, function() { $this->wp->addScreenOption('per_page', [ 'label' => _x( 'Number of newsletters per page', 'newsletters per page (screen options)', 'mailpoet' ), 'option' => 'mailpoet_newsletters_per_page', ]); }); // newsletter editor $this->wp->addSubmenuPage( self::EMAILS_PAGE_SLUG, $this->setPageTitle(__('Newsletter', 'mailpoet')), esc_html__('Newsletter Editor', 'mailpoet'), AccessControl::PERMISSION_MANAGE_EMAILS, self::EMAIL_EDITOR_PAGE_SLUG, [ $this, 'newletterEditor', ] ); // newsletter editor $this->wp->addSubmenuPage( self::EMAILS_PAGE_SLUG, $this->setPageTitle(__('Email', 'mailpoet')), esc_html__('Email Editor', 'mailpoet'), AccessControl::PERMISSION_MANAGE_EMAILS, self::EMAIL_EDITOR_V2_PAGE_SLUG, [ $this, 'emailEditor', ] ); $this->registerAutomationMenu(); // Forms page $formsPage = $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Forms', 'mailpoet')), esc_html__('Forms', 'mailpoet'), AccessControl::PERMISSION_MANAGE_FORMS, self::FORMS_PAGE_SLUG, [ $this, 'forms', ] ); // add limit per page to screen options $this->wp->addAction('load-' . $formsPage, function() { $this->wp->addScreenOption('per_page', [ 'label' => _x( 'Number of forms per page', 'forms per page (screen options)', 'mailpoet' ), 'option' => 'mailpoet_forms_per_page', ]); }); // form editor $formEditorPage = $this->wp->addSubmenuPage( self::FORMS_PAGE_SLUG, $this->setPageTitle(__('Form Editor', 'mailpoet')), esc_html__('Form Editor', 'mailpoet'), AccessControl::PERMISSION_MANAGE_FORMS, self::FORM_EDITOR_PAGE_SLUG, [ $this, 'formEditor', ] ); // add body class for form editor page $this->wp->addAction('load-' . $formEditorPage, function() { $this->wp->addFilter('admin_body_class', function ($classes) { return ltrim($classes . ' block-editor-page'); }); }); // form editor templates $formTemplateSelectionEditorPage = $this->wp->addSubmenuPage( self::FORMS_PAGE_SLUG, $this->setPageTitle(__('Select Form Template', 'mailpoet')), esc_html__('Select Form Template', 'mailpoet'), AccessControl::PERMISSION_MANAGE_FORMS, self::FORM_TEMPLATES_PAGE_SLUG, [ $this, 'formEditorTemplateSelection', ] ); // Subscribers page $subscribersPage = $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Subscribers', 'mailpoet')), esc_html__('Subscribers', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SUBSCRIBERS, self::SUBSCRIBERS_PAGE_SLUG, [ $this, 'subscribers', ] ); // add limit per page to screen options $this->wp->addAction('load-' . $subscribersPage, function() { $this->wp->addScreenOption('per_page', [ 'label' => _x( 'Number of subscribers per page', 'subscribers per page (screen options)', 'mailpoet' ), 'option' => 'mailpoet_subscribers_per_page', ]); }); // import $this->wp->addSubmenuPage( self::SUBSCRIBERS_PAGE_SLUG, $this->setPageTitle(__('Import', 'mailpoet')), esc_html__('Import', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SUBSCRIBERS, self::IMPORT_PAGE_SLUG, [ $this, 'import', ] ); // export $this->wp->addSubmenuPage( self::SUBSCRIBERS_PAGE_SLUG, $this->setPageTitle(__('Export', 'mailpoet')), esc_html__('Export', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SUBSCRIBERS, self::EXPORT_PAGE_SLUG, [ $this, 'export', ] ); // Lists page $listsPage = $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Lists', 'mailpoet')), esc_html__('Lists', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SEGMENTS, self::LISTS_PAGE_SLUG, [ $this, 'lists', ] ); // add limit per page to screen options $this->wp->addAction('load-' . $listsPage, function() { $this->wp->addScreenOption('per_page', [ 'label' => _x( 'Number of lists per page', 'lists per page (screen options)', 'mailpoet' ), 'option' => 'mailpoet_lists_per_page', ]); }); // Segments page $segmentsPage = $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Segments', 'mailpoet')), esc_html__('Segments', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SEGMENTS, self::SEGMENTS_PAGE_SLUG, [ $this, 'segments', ] ); // add limit per page to screen options $this->wp->addAction('load-' . $segmentsPage, function() { $this->wp->addScreenOption('per_page', [ 'label' => _x( 'Number of segments per page', 'segments per page (screen options)', 'mailpoet' ), 'option' => 'mailpoet_segments_per_page', ]); }); // Settings page $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Settings', 'mailpoet')), esc_html__('Settings', 'mailpoet'), AccessControl::PERMISSION_MANAGE_SETTINGS, self::SETTINGS_PAGE_SLUG, [ $this, 'settings', ] ); // Help page $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Help', 'mailpoet')), esc_html__('Help', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::HELP_PAGE_SLUG, [ $this, 'help', ] ); // Upgrade page // Only show this page in menu if the Premium plugin is not activated if (!License::getLicense()) { $this->wp->addSubmenuPage( self::MAIN_PAGE_SLUG, $this->setPageTitle(__('Upgrade', 'mailpoet')), esc_html__('Upgrade', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::UPGRADE_PAGE_SLUG, [ $this, 'upgrade', ] ); } // WooCommerce Setup $this->wp->addSubmenuPage( self::NO_PARENT_PAGE_SLUG, $this->setPageTitle(__('WooCommerce Setup', 'mailpoet')), esc_html__('WooCommerce Setup', 'mailpoet'), AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::WOOCOMMERCE_SETUP_PAGE_SLUG, [ $this, 'wooCommerceSetup', ] ); // Experimental page $this->wp->addSubmenuPage( self::SETTINGS_PAGE_SLUG, $this->setPageTitle(__('Experimental Features', 'mailpoet')), '', AccessControl::PERMISSION_MANAGE_FEATURES, self::EXPERIMENTS_PAGE_SLUG, [$this, 'experimentalFeatures'] ); // display logs page $this->wp->addSubmenuPage( self::SETTINGS_PAGE_SLUG, $this->setPageTitle(__('Logs', 'mailpoet')), '', AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, self::LOGS_PAGE_SLUG, [$this, 'logs'] ); } private function registerAutomationMenu() { $parentSlug = self::MAIN_PAGE_SLUG; // Automations menu is hidden when the subscription is part of a bundle and AutomateWoo is active but pages can be accessed directly $showAutomations = !($this->wp->isPluginActive('automatewoo/automatewoo.php') && $this->servicesChecker->isBundledSubscription()); if ( !$this->wp->applyFilters('mailpoet_show_automations', $showAutomations) ) { $parentSlug = self::NO_PARENT_PAGE_SLUG; } $automationPage = $this->wp->addSubmenuPage( $parentSlug, $this->setPageTitle(__('Automations', 'mailpoet')), // @ToDo Remove Beta once Automation is no longer beta. '' . esc_html__('Automations', 'mailpoet') . 'Beta', AccessControl::PERMISSION_MANAGE_EMAILS, self::AUTOMATIONS_PAGE_SLUG, [$this, 'automation'] ); // Automation editor $automationEditorPage = $this->wp->addSubmenuPage( self::AUTOMATIONS_PAGE_SLUG, $this->setPageTitle(__('Automation Editor', 'mailpoet')), esc_html__('Automation Editor', 'mailpoet'), AccessControl::PERMISSION_MANAGE_AUTOMATIONS, self::AUTOMATION_EDITOR_PAGE_SLUG, [$this, 'automationEditor'] ); // Automation analytics $this->wp->addSubmenuPage( self::AUTOMATIONS_PAGE_SLUG, $this->setPageTitle(__('Automation Analytics', 'mailpoet')), esc_html__('Automation Analytics', 'mailpoet'), AccessControl::PERMISSION_MANAGE_AUTOMATIONS, self::AUTOMATION_ANALYTICS_PAGE_SLUG, [$this, 'automationAnalytics'] ); // Automation templates $this->wp->addSubmenuPage( self::AUTOMATIONS_PAGE_SLUG, $this->setPageTitle(__('Automation Templates', 'mailpoet')), esc_html__('Automation Templates', 'mailpoet'), AccessControl::PERMISSION_MANAGE_AUTOMATIONS, self::AUTOMATION_TEMPLATES_PAGE_SLUG, [$this, 'automationTemplates'] ); // add body class for automation editor page $this->wp->addAction('load-' . $automationPage, function() { $this->wp->addFilter('admin_body_class', function ($classes) { return ltrim($classes . ' mailpoet-automation-is-onboarding'); }); }); $this->wp->addAction('load-' . $automationEditorPage, function() { $this->wp->addFilter('admin_body_class', function ($classes) { return ltrim($classes . ' site-editor-php'); }); }); } public function disableWPEmojis() { $this->wp->removeAction('admin_print_scripts', 'print_emoji_detection_script'); $this->wp->removeAction('admin_print_styles', 'print_emoji_styles'); } public function welcomeWizard() { $this->container->get(WelcomeWizard::class)->render(); } public function landingPage() { $this->container->get(Landingpage::class)->render(); } public function wooCommerceSetup() { $this->container->get(WooCommerceSetup::class)->render(); } public function upgrade() { $this->container->get(Upgrade::class)->render(); } public function settings() { $this->container->get(Settings::class)->render(); } public function help() { $this->container->get(Help::class)->render(); } public function homepage() { $this->container->get(Homepage::class)->render(); } public function automation() { $this->container->get(Automation::class)->render(); } public function automationTemplates() { $this->container->get(AutomationTemplates::class)->render(); } public function automationEditor() { $this->container->get(AutomationEditor::class)->render(); } public function automationAnalytics() { $this->container->get(AutomationAnalytics::class)->render(); } public function experimentalFeatures() { $this->container->get(ExperimentalFeatures::class)->render(); } public function logs() { $this->container->get(Logs::class)->render(); } public function subscribers() { $this->container->get(Subscribers::class)->render(); } public function lists() { $this->container->get(StaticSegments::class)->render(); } public function segments() { $this->container->get(DynamicSegments::class)->render(); } public function forms() { $this->container->get(Forms::class)->render(); } public function newsletters() { $this->container->get(Newsletters::class)->render(); } public function newletterEditor() { $this->container->get(NewsletterEditor::class)->render(); } public function emailEditor() { $this->container->get(EmailEditorPage::class)->render(); } public function import() { $this->container->get(SubscribersImport::class)->render(); } public function export() { $this->container->get(SubscribersExport::class)->render(); } public function formEditor() { $this->container->get(FormEditor::class)->render(); } public function formEditorTemplateSelection() { $this->container->get(FormEditor::class)->renderTemplateSelection(); } public function setPageTitle($title) { return sprintf( '%s - %s', __('MailPoet', 'mailpoet'), $title ); } public function highlightNestedMailPoetSubmenus($parentFile) { global $plugin_page, $submenu, $submenu_file; $page = $this->getPageFromContext(); if ($page) { $plugin_page = $page; return $parentFile; } if ($this->checkIsGutenbergEmailEditorPage()) { $plugin_page = self::EMAILS_PAGE_SLUG; $submenu_file = self::EMAILS_PAGE_SLUG; return self::EMAILS_PAGE_SLUG; } if ($parentFile === self::MAIN_PAGE_SLUG || !self::isOnMailPoetAdminPage()) { return $parentFile; } // find slug of the current submenu item $parentSlug = null; foreach ($submenu as $groupSlug => $group) { foreach ($group as $item) { if (($item[2] ?? null) === $plugin_page) { $parentSlug = $groupSlug; break 2; } } } if ($parentSlug && $parentSlug !== self::NO_PARENT_PAGE_SLUG) { // highlight parent submenu item $plugin_page = $parentSlug; } else { // no parent, hide MailPoet submenu for setup, wizards, error pages, etc. unset($submenu[self::MAIN_PAGE_SLUG]); $plugin_page = self::MAIN_PAGE_SLUG; } return $parentFile; } public static function isOnMailPoetAutomationPage(): bool { $screenId = isset($_REQUEST['page']) ? sanitize_text_field(wp_unslash($_REQUEST['page'])) : ''; $automationPages = [ 'mailpoet-automation', 'mailpoet-automation-templates', 'mailpoet-automation-editor', ]; return in_array( $screenId, $automationPages, true ); } public static function isOnMailPoetAdminPage(array $exclude = null, $screenId = null) { if (is_null($screenId)) { if (empty($_REQUEST['page'])) { return false; } $screenId = sanitize_text_field(wp_unslash($_REQUEST['page'])); } if (!empty($exclude)) { foreach ($exclude as $slug) { if (stripos($screenId, $slug) !== false) { return false; } } } return (stripos($screenId, 'mailpoet-') !== false); } /** * This error page is used when the initialization is failed * to display admin notices only */ public static function addErrorPage(AccessControl $accessControl) { if (!self::isOnMailPoetAdminPage() || !isset($_REQUEST['page'])) { return false; } $page = sanitize_text_field(wp_unslash($_REQUEST['page'])); // Check if page already exists if ( get_plugin_page_hook($page, '') || WPFunctions::get()->getPluginPageHook($page, self::MAIN_PAGE_SLUG) ) { return false; } WPFunctions::get()->addSubmenuPage( self::NO_PARENT_PAGE_SLUG, 'MailPoet', 'MailPoet', AccessControl::PERMISSION_ACCESS_PLUGIN_ADMIN, $page, [ __CLASS__, 'errorPageCallback', ] ); } public static function errorPageCallback() { // Used for displaying admin notices only } public function checkPremiumKey(ServicesChecker $checker = null) { $showNotices = isset($_SERVER['SCRIPT_NAME']) && stripos(sanitize_text_field(wp_unslash($_SERVER['SCRIPT_NAME'])), 'plugins.php') !== false; $checker = $checker ?: $this->servicesChecker; $this->premiumKeyValid = $checker->isPremiumKeyValid($showNotices); } public function getPageFromContext(): ?string { $context = isset($_GET['context']) ? sanitize_text_field(wp_unslash($_GET['context'])) : null; if ($context === 'automation') { return self::AUTOMATIONS_PAGE_SLUG; } return null; } private function checkIsGutenbergEmailEditorPage(): bool { return $this->wp->getPostType() === EmailEditor::MAILPOET_EMAIL_POST_TYPE; } }