diff --git a/assets/css/src/box.styl b/assets/css/src/box.styl index 5202d0b5ac..78c97a39b8 100644 --- a/assets/css/src/box.styl +++ b/assets/css/src/box.styl @@ -71,15 +71,17 @@ $box-description-font-size = $box-description-line-height .mailpoet_boxes .mailpoet_description float:left - width: 245px + width: 258px max-height: $box-description-height padding-bottom: 0 overflow: hidden + word-wrap: break-word + overflow-wrap: break-word h3 margin: 0 0 $box-description-space-between-heading-and-paragraph 0 overflow: hidden - max-width: 210px + max-width: 223px line-height: $box-heading-line-height font-size: $box-heading-font-size @@ -117,6 +119,20 @@ $box-description-font-size = $box-description-line-height [data-type="standard"] .mailpoet_thumbnail background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20version%3D%221.1%22%20id%3D%22Layer_1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%20x%3D%220px%22%20y%3D%220px%22%20viewBox%3D%220%200%2070%2070%22%20enable-background%3D%22new%200%200%2070%2070%22%20xml%3Aspace%3D%22preserve%22%3E%3Cpath%20fill%3D%22%23ffffff%22%20fill-rule%3D%22evenodd%22%20clip-rule%3D%22evenodd%22%20d%3D%22M62.751%2C65.368L43.923%2C50.955l18.832-14.377l-5.395-4.118v-3.479l5.983%2C4.568%20v3.479l0.166%2C26.269C63.51%2C64.089%2C63.198%2C64.783%2C62.751%2C65.368z%20M31.433%2C49.5l-15.955-12v-30c0-1.657%2C1.339-3%2C2.992-3h33.905%20c1.652%2C0%2C2.992%2C1.343%2C2.992%2C3v30l-15.955%2C12H31.433z%20M18.469%2C35.5H33.5v-2H18.469V35.5z%20M18.469%2C31.5H33.5v-2H18.469V31.5z%20M18.469%2C27.5H33.5v-2H18.469V27.5z%20M38.406%2C7.5H18.469v14h19.937V7.5z%20M52.375%2C7.5H40.408v2h11.967V7.5z%20M52.375%2C11.5H40.408v2%20h11.967V11.5z%20M52.375%2C15.5H40.408v2h11.967V15.5z%20M52.375%2C19.5H40.408v2h11.967V19.5z%20M52.375%2C25.5H37.5v2h14.875V25.5z%20M52.375%2C29.5H37.5v2h14.875V29.5z%20M52.375%2C33.5H37.5v2h14.875V33.5z%20M26.14%2C17.192L28.442%2C9.5l3.277%2C6.571l1.71-2.571l2.992%2C6%20h-2.992h-3.989h-0.997H25.45h-4.986l3.989-4L26.14%2C17.192z%20M22.469%2C12.5h-1c-0.552%2C0-1-0.448-1-1v-1c0-0.552%2C0.448-1%2C1-1h1%20c0.552%2C0%2C1%2C0.448%2C1%2C1v1C23.469%2C12.052%2C23.021%2C12.5%2C22.469%2C12.5z%20M26.795%2C50.955L8.05%2C65.305c-0.419-0.574-0.55-1.41-0.55-2.174%20V37.015l-0.015%2C0.011v-3.479l5.998-4.579v3.479L8.017%2C36.62L26.795%2C50.955z%20M29.137%2C52.659v-0.157h12.462v0.144l0.047-0.036%20L59.73%2C66.5H10.989l18.084-13.889L29.137%2C52.659z%22%2F%3E%3C%2Fsvg%3E%0A") +[data-type="woocommerce"] .mailpoet_thumbnail + background-image: url('') + .mailpoet_boxes_preview margin: -10px padding: 10px 20px + +.mailpoet_boxes + .title_and_badge + display: flex + flex-direction: row + justify-content: space-between + + .mailpoet_badge + margin: 0 0 0 10px + padding: 0 6px 0 6px + max-height: 21px \ No newline at end of file diff --git a/assets/css/src/listing/newsletters.styl b/assets/css/src/listing/newsletters.styl index 10c2f8e86f..e8b68970ae 100644 --- a/assets/css/src/listing/newsletters.styl +++ b/assets/css/src/listing/newsletters.styl @@ -41,13 +41,13 @@ $green-badge-color = #55bd56 vertical-align: middle white-space: nowrap - &_excellent + &_excellent, &_teal background: $excellent-badge-color - &_good + &_good, &_yellow background: $good-badge-color - &_bad + &_bad, &_red background: $bad-badge-color &_green diff --git a/assets/js/src/newsletters/newsletters.jsx b/assets/js/src/newsletters/newsletters.jsx index 7eeea9dcfe..ecfb5d1923 100644 --- a/assets/js/src/newsletters/newsletters.jsx +++ b/assets/js/src/newsletters/newsletters.jsx @@ -3,14 +3,14 @@ import ReactDOM from 'react-dom'; import { Router, Route, IndexRedirect, useRouterHistory } from 'react-router'; import { createHashHistory } from 'history'; import Hooks from 'wp-js-hooks'; +import _ from 'underscore'; import NewsletterTypes from 'newsletters/types.jsx'; import NewsletterTemplates from 'newsletters/templates.jsx'; import NewsletterSend from 'newsletters/send.jsx'; - import NewsletterTypeStandard from 'newsletters/types/standard.jsx'; import NewsletterTypeNotification from 'newsletters/types/notification/notification.jsx'; - +import AutomaticEmailsEventsList from 'newsletters/types/automatic_emails/events_list.jsx'; import NewsletterListStandard from 'newsletters/listings/standard.jsx'; import NewsletterListWelcome from 'newsletters/listings/welcome.jsx'; import NewsletterListNotification from 'newsletters/listings/notification.jsx'; @@ -26,33 +26,82 @@ const App = React.createClass({ const container = document.getElementById('newsletters_container'); +const getAutomaticEmailsRoutes = () => { + if (!window.mailpoet_automatic_emails) return null; + + return _.map(window.mailpoet_automatic_emails, automaticEmail => ({ + path: `new/${automaticEmail.id}`, + name: automaticEmail.id, + component: AutomaticEmailsEventsList, + data: { + automaticEmail: automaticEmail, + }, + })); +}; + if (container) { - let extraRoutes = []; - extraRoutes = Hooks.applyFilters('mailpoet_newsletters_before_router', extraRoutes); + let routes = [ + /* Listings */ + { + path: 'standard(/)**', + params: { tab: 'standard' }, + component: NewsletterListStandard, + }, + { + path: 'welcome(/)**', + component: NewsletterListWelcome, + }, + { + path: 'notification/history/:parent_id(/)**', + component: NewsletterListNotificationHistory, + }, + { + path: 'notification(/)**', + component: NewsletterListNotification, + }, + /* Newsletter: type selection */ + { + path: 'new', + component: NewsletterTypes, + }, + /* New newsletter: types */ + { + path: 'new/standard', + component: NewsletterTypeStandard, + }, + { + path: 'new/notification', + component: NewsletterTypeNotification, + }, + /* Template selection */ + { + name: 'template', + path: 'template/:id', + component: NewsletterTemplates, + }, + /* Sending options */ + { + path: 'send/:id', + component: NewsletterSend, + }, + ]; + + routes = Hooks.applyFilters('mailpoet_newsletters_before_router', [...routes, ...getAutomaticEmailsRoutes()]); const mailpoetListing = ReactDOM.render(( - {/* Listings */} - - - - - {/* Newsletter: type selection */} - - {/* New newsletter: types */} - - - {/* Template selection */} - - {/* Sending options */} - - {/* Extra routes */} - {extraRoutes.map(rt => - - )} + {routes.map(route => ( + + ))} ), container); diff --git a/assets/js/src/newsletters/types.jsx b/assets/js/src/newsletters/types.jsx index 4c7cc38414..0bc675526a 100644 --- a/assets/js/src/newsletters/types.jsx +++ b/assets/js/src/newsletters/types.jsx @@ -5,13 +5,15 @@ define( 'wp-js-hooks', 'react-router', 'newsletters/breadcrumb.jsx', + 'underscore', ], ( React, MailPoet, Hooks, Router, - Breadcrumb + Breadcrumb, + _ ) => { const NewsletterTypes = React.createClass({ contextTypes: { @@ -50,8 +52,29 @@ define( } }); }, + getAutomaticEmails: function () { + if (!window.mailpoet_automatic_emails) return []; + + return _.map(window.mailpoet_automatic_emails, (automaticEmail) => { + const email = automaticEmail; + const disabled = !email.events; + + email.action = (() => ( +
+ + { MailPoet.I18n.t('setUp') } + +
+ ))(); + + return email; + }); + }, render: function () { - let types = [ + const defaultTypes = [ { id: 'standard', title: MailPoet.I18n.t('regularNewsletterTypeTitle'), @@ -72,7 +95,7 @@ define( return (
- {MailPoet.I18n.t('getPremiumVersion')} + {MailPoet.I18n.t('premiumFeatureLink')}
); @@ -92,7 +115,7 @@ define( }, ]; - types = Hooks.applyFilters('mailpoet_newsletters_types', types, this); + const types = Hooks.applyFilters('mailpoet_newsletters_types', [...defaultTypes, ...this.getAutomaticEmails()], this); return (
diff --git a/assets/js/src/newsletters/types/automatic_emails/breadcrumb.jsx b/assets/js/src/newsletters/types/automatic_emails/breadcrumb.jsx new file mode 100644 index 0000000000..6a4a247c1f --- /dev/null +++ b/assets/js/src/newsletters/types/automatic_emails/breadcrumb.jsx @@ -0,0 +1,41 @@ +import Breadcrumb from 'newsletters/breadcrumb.jsx'; +import React from 'react'; +import MailPoet from 'mailpoet'; + +class AutomaticEmailsBreadcrumb extends React.Component { + render() { + const steps = [ + { + name: 'type', + label: MailPoet.I18n.t('selectType'), + link: '/new', + }, + { + name: 'events', + label: MailPoet.I18n.t('events'), + }, + { + name: 'conditions', + label: MailPoet.I18n.t('conditions'), + }, + { + name: 'template', + label: MailPoet.I18n.t('template'), + }, + { + name: 'editor', + label: MailPoet.I18n.t('designer'), + }, + { + name: 'send', + label: MailPoet.I18n.t('send'), + }, + ]; + + return ( + + ); + } +} + +module.exports = AutomaticEmailsBreadcrumb; diff --git a/assets/js/src/newsletters/types/automatic_emails/events_list.jsx b/assets/js/src/newsletters/types/automatic_emails/events_list.jsx new file mode 100644 index 0000000000..f68f4a07e9 --- /dev/null +++ b/assets/js/src/newsletters/types/automatic_emails/events_list.jsx @@ -0,0 +1,93 @@ +import React from 'react'; +import AutomaticEmailsBreadcrumb from 'newsletters/types/automatic_emails/breadcrumb.jsx'; +import MailPoet from 'mailpoet'; +import _ from 'underscore'; + +class AutomaticEmailsEventsList extends React.Component { + constructor(props) { + super(props); + this.automaticEmail = this.props.route.data.automaticEmail; + this.automaticEmailEvents = this.automaticEmail.events; + this.eventsConfigurator = this.eventsConfigurator.bind(this); + } + + eventsConfigurator(eventId) { + this.props.router.push(`new/${this.automaticEmail.id}/${eventId}/conditions`); + } + + displayEvents() { + const events = _.map(this.automaticEmailEvents, (event, index) => { + let action; + + if (this.automaticEmail.premium) { + action = ( + + {MailPoet.I18n.t('premiumFeatureLink')} + + ); + } else { + const disabled = event.soon; + + action = ( + + {event.actionButtonTitle || MailPoet.I18n.t('setUp')} + + ); + } + + return ( +
  • +
    +
    + {event.thumbnailImage ? : null} +
    +
    +
    +

    {event.title} {event.soon ? `(${MailPoet.I18n.t('soon').toLowerCase()})` : ''}

    + {event.badge ? ( + + {event.badge.text} + + ) : '' + } +
    +

    {event.description}

    +
    +
    + {action} +
    +
    +
  • + ); + }); + + return ( + + ); + } + + render() { + const heading = MailPoet.I18n.t('selectAutomaticEmailsEventsHeading') + .replace('%1s', this.automaticEmail.title); + + return ( +
    +

    + {heading} ({MailPoet.I18n.t('beta').toLowerCase()}) +

    + + + + {this.displayEvents()} +
    + ); + } +} + +module.exports = AutomaticEmailsEventsList; diff --git a/assets/js/src/newsletters/types/woocommerce.jsx b/assets/js/src/newsletters/types/woocommerce.jsx new file mode 100644 index 0000000000..c2d19f4d1e --- /dev/null +++ b/assets/js/src/newsletters/types/woocommerce.jsx @@ -0,0 +1,161 @@ +define( + [ + 'react', + 'mailpoet', + 'wp-js-hooks', + 'react-router', + 'newsletters/breadcrumb.jsx', + ], + ( + React, + MailPoet, + Hooks, + Router, + Breadcrumb + ) => { + const WooCommerceAutomaticEmail = React.createClass({ + contextTypes: { + router: React.PropTypes.object.isRequired, + }, + setupNewsletter: function (type) { + if (type !== undefined) { + this.context.router.push(`/new/${type}`); + MailPoet.trackEvent('Emails > Type selected', { + 'MailPoet Free version': window.mailpoet_version, + 'Email type': type, + }); + } + }, + createNewsletter: function (type) { + MailPoet.trackEvent('Emails > Type selected', { + 'MailPoet Free version': window.mailpoet_version, + 'Email type': type, + }); + MailPoet.Ajax.post({ + api_version: window.mailpoet_api_version, + endpoint: 'newsletters', + action: 'create', + data: { + type: type, + subject: MailPoet.I18n.t('draftNewsletterTitle'), + }, + }).done((response) => { + this.context.router.push(`/template/${response.data.id}`); + }).fail((response) => { + if (response.errors.length > 0) { + MailPoet.Notice.error( + response.errors.map(error => error.message), + { scroll: true } + ); + } + }); + }, + render: function () { + const types = [ + { + id: 'woocommerce', + title: MailPoet.I18n.t('wooCommerceEventAbandonedCartTitle'), + description: MailPoet.I18n.t('wooCommerceEventAbandonedCartDescription'), + badge: { + text: MailPoet.I18n.t('wooCommerceEventAbandonedCartBadge'), + style: 'red', + }, + }, + { + id: 'woocommerce', + title: MailPoet.I18n.t('wooCommerceEventFirstPurchaseTitle'), + description: MailPoet.I18n.t('wooCommerceEventFirstPurchaseDescription'), + badge: { + text: MailPoet.I18n.t('wooCommerceEventFirstPurchaseBadge'), + style: 'yellow', + }, + }, + { + id: 'woocommerce', + title: MailPoet.I18n.t('wooCommerceEventPurchasedProductTitle'), + description: MailPoet.I18n.t('wooCommerceEventPurchasedProductDescription'), + }, + { + id: 'woocommerce', + title: MailPoet.I18n.t('wooCommerceEventPurchasedInCategoryTitle'), + description: MailPoet.I18n.t('wooCommerceEventPurchasedInCategoryDescription'), + soon: true, + }, + { + id: 'woocommerce', + title: MailPoet.I18n.t('wooCommerceEventBigSpenderTitle'), + description: MailPoet.I18n.t('wooCommerceEventBigSpenderDescription'), + soon: true, + badge: { + text: MailPoet.I18n.t('wooCommerceEventSmartToHaveBadge'), + style: 'teal', + }, + }, + ]; + + const steps = [ + { + name: 'type', + label: MailPoet.I18n.t('selectType'), + link: '/new', + }, + { + name: 'events', + label: MailPoet.I18n.t('wooCommerceBreadcrumbsEvents'), + }, + { + name: 'conditions', + label: MailPoet.I18n.t('wooCommerceBreadcrumbsConditions'), + }, + { + name: 'template', + label: MailPoet.I18n.t('template'), + }, + { + name: 'editor', + label: MailPoet.I18n.t('designer'), + }, + { + name: 'send', + label: MailPoet.I18n.t('send'), + }, + ]; + + return ( +
    +

    {MailPoet.I18n.t('wooCommerceSelectEventHeading')}

    + + + +
      + {types.map((type, index) => ( +
    • +
      +
      + {type.thumbnailImage ? : null} +
      +
      +
      +

      {type.title} {type.soon ? `(${MailPoet.I18n.t('wooCommerceEventTitleSoon')})` : ''}

      + {type.badge ? {type.badge.text} : ''} +
      +

      {type.description}

      +
      + + +
      +
    • + ), this)} +
    +
    + ); + }, + }); + + return WooCommerceAutomaticEmail; + } +); diff --git a/lib/Config/Menu.php b/lib/Config/Menu.php index 6534ab17f5..74c4a9933b 100644 --- a/lib/Config/Menu.php +++ b/lib/Config/Menu.php @@ -573,6 +573,56 @@ class Menu { $data['tracking_enabled'] = Setting::getValue('tracking.enabled'); $data['premium_plugin_active'] = License::getLicense(); + $data['automatic_emails'] = array( + array( + 'id' => 'woocommerce', + 'premium' => true, + 'title' => __('WooCommerce Automatic Emails', 'mailpoet'), + 'description' => __("Put your store's email marketing on autopilot.", 'mailpoet'), + 'events' => array( + array( + 'id' => 'woocommerce', + 'title' => __('Abandoned Shopping Cart', 'mailpoet'), + 'description' => __('Send an email to logged-in visitors who have items in their shopping carts but left your website without checking out. Can convert up to 5% of abandoned carts.', 'mailpoet'), + 'badge' => array( + 'text' => __('Must-have', 'mailpoet'), + 'style' => 'red' + ) + ), + array( + 'id' => 'woocommerce', + 'title' => __('First Purchase', 'mailpoet'), + 'description' => __('Send an email automatically to customers who have purchased for the first time.', 'mailpoet'), + 'badge' => array( + 'text' => __('Recommended', 'mailpoet'), + 'style' => 'yellow' + ) + ), + array( + 'id' => 'woocommerce', + 'title' => __('Purchased This Product', 'mailpoet'), + 'description' => __('Let MailPoet send an email to customers who purchase a specific product.', 'mailpoet'), + ), + array( + 'id' => 'woocommerce', + 'title' => __('Purchased In This Category', 'mailpoet'), + 'description' => __('Let MailPoet send an email to customers who purchase a product from a specific category.', 'mailpoet'), + 'soon' => true + ), + array( + 'id' => 'woocommerce', + 'title' => __('Big Spender', 'mailpoet'), + 'description' => __('Let MailPoet send an email to customers who have spent a certain amount to thank them, possibly with a coupon.', 'mailpoet'), + 'soon' => true, + 'badge' => array( + 'text' => __('Smart to have', 'mailpoet'), + 'style' => 'teal' + ) + ) + ) + ) + ); + wp_enqueue_script('jquery-ui'); wp_enqueue_script('jquery-ui-datepicker'); diff --git a/views/newsletters.html b/views/newsletters.html index de28b3177c..b4cc597b8f 100644 --- a/views/newsletters.html +++ b/views/newsletters.html @@ -16,6 +16,7 @@ var mailpoet_date_storage_format = "Y-m-d"; var mailpoet_tracking_enabled = <%= json_encode(tracking_enabled) %>; var mailpoet_premium_active = <%= json_encode(premium_plugin_active) %>; + var mailpoet_automatic_emails = <%= json_encode(automatic_emails) %>; <% endblock %> @@ -55,6 +56,8 @@ 'numberOfItemsMultiple': __('%$1d items'), 'selectType': __('Select type'), + 'events': __('Events'), + 'conditions': _x('Conditions', 'Configuration options for automatic email events'), 'template': __('Template'), 'designer': __('Designer'), 'send': __('Send'), @@ -125,7 +128,7 @@ 'create': __('Create'), 'welcomeNewsletterTypeTitle': __('Welcome Email'), 'welcomeNewsletterTypeDescription': __('Automatically send an email (or series of emails) to new subscribers or WordPress users. Send a day, a week, or a month after they sign up.'), - 'getPremiumVersion': __('Get premium version!'), + 'premiumFeatureLink': __('This is a Premium feature'), 'setUp': __('Set up'), 'postNotificationNewsletterTypeTitle': __('Latest Post Notifications'), 'postNotificationNewsletterTypeDescription': __('Let MailPoet email your subscribers with your latest content. You can send daily, weekly, monthly, or even immediately after publication.'), @@ -272,7 +275,10 @@ 'sample': _x('Sample', 'Sample newsletters templates category'), 'noTemplates': __('This category does not contain any template yet!'), - 'errorWhileTakingScreenshot': __('An error occured while saving the template in "Recently sent"') + 'soon': __('Soon'), + 'beta': __('Beta'), + 'errorWhileTakingScreenshot': __('An error occured while saving the template in "Recently sent"'), + 'selectAutomaticEmailsEventsHeading': __('Select %1s events'), }) %> <% endblock %> diff --git a/webpack.config.js b/webpack.config.js index c0245ec09b..2eccd90c36 100644 --- a/webpack.config.js +++ b/webpack.config.js @@ -129,6 +129,14 @@ var baseConfig = { include: path.resolve(__dirname, 'assets/js/src/newsletters/breadcrumb.jsx'), loader: 'expose-loader?' + globalPrefix + '.NewsletterCreationBreadcrumb!babel-loader', }, + { + include: path.resolve(__dirname, 'assets/js/src/newsletters/types/automatic_emails/events_list.jsx'), + loader: 'expose-loader?' + globalPrefix + '.AutomaticEmailsEventsList!babel-loader', + }, + { + include: path.resolve(__dirname, 'assets/js/src/newsletters/types/automatic_emails/breadcrumb.jsx'), + loader: 'expose-loader?' + globalPrefix + '.AutomaticEmailsBreadcrumb!babel-loader', + }, { include: /Blob.js$/, loader: 'exports-loader?window.Blob', @@ -185,6 +193,8 @@ var adminConfig = { 'form/form.jsx', 'newsletters/badges/stats.jsx', 'newsletters/breadcrumb.jsx', + 'newsletters/types/automatic_emails/events_list.jsx', + 'newsletters/types/automatic_emails/breadcrumb.jsx', 'newsletters/types/welcome/scheduling.jsx', 'history', ],