diff --git a/projects/js-packages/components/changelog/add-growth-upsells-to-social-and-stats-interstials b/projects/js-packages/components/changelog/add-growth-upsells-to-social-and-stats-interstials new file mode 100644 index 0000000000000..dda6e2d1a89aa --- /dev/null +++ b/projects/js-packages/components/changelog/add-growth-upsells-to-social-and-stats-interstials @@ -0,0 +1,4 @@ +Significance: patch +Type: added + +Add Stats icon diff --git a/projects/js-packages/components/components/icons/index.tsx b/projects/js-packages/components/components/icons/index.tsx index a390b8a3a3f6f..892e03dee4997 100644 --- a/projects/js-packages/components/components/icons/index.tsx +++ b/projects/js-packages/components/components/icons/index.tsx @@ -234,6 +234,18 @@ export const AiIcon: React.FC< BaseIconProps > = ( { ); }; +export const StatsIcon: React.FC< BaseIconProps > = ( { opacity = 1, size, color } ) => { + return ( + + + + ); +}; + const jetpackIcons = { 'anti-spam': AntiSpamIcon, backup: BackupIcon, @@ -249,6 +261,7 @@ const jetpackIcons = { jetpack: JetpackIcon, share: ShareIcon, ai: AiIcon, + stats: StatsIcon, }; const iconsMap = { diff --git a/projects/packages/my-jetpack/_inc/components/product-cards-section/all.ts b/projects/packages/my-jetpack/_inc/components/product-cards-section/all.ts index 10e29b7fbca76..880d6e7ea0a21 100644 --- a/projects/packages/my-jetpack/_inc/components/product-cards-section/all.ts +++ b/projects/packages/my-jetpack/_inc/components/product-cards-section/all.ts @@ -28,4 +28,5 @@ export const JetpackModuleToProductCard: { scan: null, security: null, creator: null, + growth: null, }; diff --git a/projects/packages/my-jetpack/_inc/components/product-cards-section/index.tsx b/projects/packages/my-jetpack/_inc/components/product-cards-section/index.tsx index ff10798442fc2..855ba5e234b12 100644 --- a/projects/packages/my-jetpack/_inc/components/product-cards-section/index.tsx +++ b/projects/packages/my-jetpack/_inc/components/product-cards-section/index.tsx @@ -26,7 +26,7 @@ type DisplayItemType = Record< // We don't have a card for Security or Extras, and scan is displayed as protect. // 'jetpack-ai' is the official slug for the AI module, so we also exclude 'ai'. // The backend still supports the 'ai' slug, so it is part of the JetpackModule type. - Exclude< JetpackModule, 'extras' | 'scan' | 'security' | 'ai' | 'creator' >, + Exclude< JetpackModule, 'extras' | 'scan' | 'security' | 'ai' | 'creator' | 'growth' >, FC< { admin: boolean } > >; @@ -104,7 +104,12 @@ const ProductCardsSection: FC< ProductCardsSectionProps > = ( { noticeMessage } const filterProducts = ( products: JetpackModule[] ) => { return products.filter( product => { - if ( product === 'scan' || product === 'security' || product === 'extras' ) { + if ( + product === 'scan' || + product === 'security' || + product === 'growth' || + product === 'extras' + ) { return false; } return true; diff --git a/projects/packages/my-jetpack/_inc/components/product-interstitial/index.jsx b/projects/packages/my-jetpack/_inc/components/product-interstitial/index.jsx index 60cd848f1f817..a2da42c2af781 100644 --- a/projects/packages/my-jetpack/_inc/components/product-interstitial/index.jsx +++ b/projects/packages/my-jetpack/_inc/components/product-interstitial/index.jsx @@ -32,8 +32,6 @@ import boostImage from './boost.png'; import crmImage from './crm.png'; import extrasImage from './extras.png'; import searchImage from './search.png'; -import socialImage from './social.png'; -import statsImage from './stats.png'; import styles from './style.module.scss'; import videoPressImage from './videopress.png'; @@ -408,17 +406,7 @@ export function ScanInterstitial() { * @return {object} SocialInterstitial react component. */ export function SocialInterstitial() { - return ( - - { - - ); + return ; } /** @@ -462,15 +450,8 @@ export function StatsInterstitial() { directCheckout={ true } installsPlugin={ true } ctaButtonLabel={ __( 'Get Stats', 'jetpack-my-jetpack' ) } - > - { - + bundle="growth" + /> ); } diff --git a/projects/packages/my-jetpack/_inc/data/constants.ts b/projects/packages/my-jetpack/_inc/data/constants.ts index c11217819409d..c55a4a6b24f61 100644 --- a/projects/packages/my-jetpack/_inc/data/constants.ts +++ b/projects/packages/my-jetpack/_inc/data/constants.ts @@ -51,4 +51,5 @@ export const PRODUCT_SLUGS = { PROTECT: 'protect', VIDEOPRESS: 'videopress', STATS: 'stats', + GROWTH: 'growth', } satisfies Record< string, JetpackModule >; diff --git a/projects/packages/my-jetpack/changelog/add-growth-upsells-to-social-and-stats-interstials b/projects/packages/my-jetpack/changelog/add-growth-upsells-to-social-and-stats-interstials new file mode 100644 index 0000000000000..5d0c9263fc127 --- /dev/null +++ b/projects/packages/my-jetpack/changelog/add-growth-upsells-to-social-and-stats-interstials @@ -0,0 +1,4 @@ +Significance: patch +Type: added + +Add growth upsell to Stats and Social interstitials diff --git a/projects/packages/my-jetpack/global.d.ts b/projects/packages/my-jetpack/global.d.ts index 49d4563fabc82..ec2870487c89d 100644 --- a/projects/packages/my-jetpack/global.d.ts +++ b/projects/packages/my-jetpack/global.d.ts @@ -39,7 +39,8 @@ type JetpackModule = | 'security' | 'protect' | 'videopress' - | 'stats'; + | 'stats' + | 'growth'; type ThreatItem = { // Protect API properties (free plan) diff --git a/projects/packages/my-jetpack/src/class-products.php b/projects/packages/my-jetpack/src/class-products.php index dcd3fcc28b3cd..7d17c464804cd 100644 --- a/projects/packages/my-jetpack/src/class-products.php +++ b/projects/packages/my-jetpack/src/class-products.php @@ -120,6 +120,7 @@ public static function get_products_classes() { 'videopress' => Products\Videopress::class, 'stats' => Products\Stats::class, 'ai' => Products\Jetpack_Ai::class, + 'growth' => Products\Growth::class, ); /** diff --git a/projects/packages/my-jetpack/src/products/class-growth.php b/projects/packages/my-jetpack/src/products/class-growth.php new file mode 100644 index 0000000000000..7478574449aa0 --- /dev/null +++ b/projects/packages/my-jetpack/src/products/class-growth.php @@ -0,0 +1,209 @@ + true, + 'wpcom_product_slug' => $product_slug, + ), + Wpcom_Products::get_product_pricing( $product_slug ) + ); + } + + /** + * Get the WPCOM product slug used to make the purchase + * + * @return string + */ + public static function get_wpcom_product_slug() { + return 'jetpack_growth_yearly'; + } + + /** + * Checks whether the Jetpack module is active + * + * This is a bundle and not a product. We should not use this information for anything + * + * @return bool + */ + public static function is_module_active() { + return false; + } + + /** + * Activates the product by installing and activating its plugin + * + * @param WP_Error|bool $current_result Is the result of the top level activation actions. You probably won't do anything if it is an WP_Error. + * @return bool|\WP_Error + */ + public static function do_product_specific_activation( $current_result ) { + $product_activation = parent::do_product_specific_activation( $current_result ); + + // A bundle is not a module. There's nothing in the plugin to be activated, so it's ok to fail to activate the module. + if ( is_wp_error( $product_activation ) && 'module_activation_failed' === $product_activation->get_error_code() ) { + return $product_activation; + } + + // At this point, Jetpack plugin is installed. Let's activate each individual product. + $activation = Social::activate(); + if ( is_wp_error( $activation ) ) { + return $activation; + } + + $activation = Stats::activate(); + if ( is_wp_error( $activation ) ) { + return $activation; + } + + return $activation; + } + + /** + * Checks whether the Product is active + * + * Growth is a bundle and not a module. Activation takes place on WPCOM. So lets consider it active if jetpack is active and has the plan. + * + * @return bool + */ + public static function is_active() { + return static::is_jetpack_plugin_active() && static::has_required_plan(); + } + + /** + * Checks whether the current plan (or purchase) of the site already supports the product + * + * @return bool + */ + public static function has_required_plan() { + $purchases_data = Wpcom_Products::get_site_current_purchases(); + if ( is_wp_error( $purchases_data ) ) { + return false; + } + if ( is_array( $purchases_data ) && ! empty( $purchases_data ) ) { + foreach ( $purchases_data as $purchase ) { + if ( + str_starts_with( $purchase->product_slug, 'jetpack_growth' ) || + str_starts_with( $purchase->product_slug, 'jetpack_complete' ) + ) { + return true; + } + } + } + return false; + } + + /** + * Checks whether the product is a bundle + * + * @return bool + */ + public static function is_bundle_product() { + return true; + } + + /** + * Returns all products it contains. + * + * @return array Product slugs + */ + public static function get_supported_products() { + return array( 'social', 'stats' ); + } + + /** + * Get the URL where the user manages the product + * + * @return string + */ + public static function get_manage_url() { + return ''; + } +} diff --git a/projects/packages/my-jetpack/src/products/class-security.php b/projects/packages/my-jetpack/src/products/class-security.php index 866899cb81223..7e7005ea9d034 100644 --- a/projects/packages/my-jetpack/src/products/class-security.php +++ b/projects/packages/my-jetpack/src/products/class-security.php @@ -69,7 +69,7 @@ public static function get_long_description() { /** * Get the internationalized features list * - * @return array Boost features list + * @return array Security features list */ public static function get_features() { return array( @@ -81,17 +81,18 @@ public static function get_features() { } /** - * Get the product princing details + * Get the product pricing details * * @return array Pricing details */ public static function get_pricing_for_ui() { + $product_slug = static::get_wpcom_product_slug(); return array_merge( array( 'available' => true, - 'wpcom_product_slug' => static::get_wpcom_product_slug(), + 'wpcom_product_slug' => $product_slug, ), - Wpcom_Products::get_product_pricing( static::get_wpcom_product_slug() ) + Wpcom_Products::get_product_pricing( $product_slug ) ); } @@ -195,7 +196,7 @@ public static function is_bundle_product() { /** * Return all the products it contains. * - * @return Array Product slugs + * @return array Product slugs */ public static function get_supported_products() { return array( 'backup', 'scan', 'anti-spam' ); diff --git a/projects/packages/my-jetpack/src/products/class-social.php b/projects/packages/my-jetpack/src/products/class-social.php index 317eeaf303a01..ae1d3ee6f76c0 100644 --- a/projects/packages/my-jetpack/src/products/class-social.php +++ b/projects/packages/my-jetpack/src/products/class-social.php @@ -197,4 +197,14 @@ public static function get_manage_url() { return admin_url( 'admin.php?page=jetpack#/settings?term=publicize' ); } + + /** + * Return product bundles list + * that supports the product. + * + * @return boolean|array Products bundle list. + */ + public static function is_upgradable_by_bundle() { + return array( 'growth' ); + } } diff --git a/projects/packages/my-jetpack/src/products/class-stats.php b/projects/packages/my-jetpack/src/products/class-stats.php index fa9aaf61f5561..c361bd6737f39 100644 --- a/projects/packages/my-jetpack/src/products/class-stats.php +++ b/projects/packages/my-jetpack/src/products/class-stats.php @@ -314,4 +314,14 @@ public static function get_purchase_url() { public static function get_manage_url() { return admin_url( 'admin.php?page=stats' ); } + + /** + * Return product bundles list + * that supports the product. + * + * @return boolean|array Products bundle list. + */ + public static function is_upgradable_by_bundle() { + return array( 'growth' ); + } } diff --git a/projects/plugins/wpcomsh/changelog/renovate-automattic-custom-fonts-3.x-lockfile b/projects/plugins/wpcomsh/changelog/renovate-automattic-custom-fonts-3.x-lockfile new file mode 100644 index 0000000000000..c47cb18e82997 --- /dev/null +++ b/projects/plugins/wpcomsh/changelog/renovate-automattic-custom-fonts-3.x-lockfile @@ -0,0 +1,4 @@ +Significance: patch +Type: changed + +Updated package dependencies. diff --git a/projects/plugins/wpcomsh/composer.lock b/projects/plugins/wpcomsh/composer.lock index 901e1573a5401..0803e22af3189 100644 --- a/projects/plugins/wpcomsh/composer.lock +++ b/projects/plugins/wpcomsh/composer.lock @@ -28,16 +28,16 @@ }, { "name": "automattic/custom-fonts", - "version": "v3.0.5", + "version": "v3.0.6", "source": { "type": "git", "url": "https://github.com/Automattic/custom-fonts.git", - "reference": "e9236c34a6ecbee3bb9d2bebe698f086dd339258" + "reference": "63d0a9251a252b74ab0862ac00f4d4d4c1d6b320" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/Automattic/custom-fonts/zipball/e9236c34a6ecbee3bb9d2bebe698f086dd339258", - "reference": "e9236c34a6ecbee3bb9d2bebe698f086dd339258", + "url": "https://api.github.com/repos/Automattic/custom-fonts/zipball/63d0a9251a252b74ab0862ac00f4d4d4c1d6b320", + "reference": "63d0a9251a252b74ab0862ac00f4d4d4c1d6b320", "shasum": "" }, "require-dev": { @@ -49,7 +49,7 @@ "GPL-2.0-or-later" ], "description": "A provider-agnostic WordPress plugin for changing theme fonts.", - "time": "2024-06-26T18:47:13+00:00" + "time": "2024-11-19T20:05:23+00:00" }, { "name": "automattic/custom-fonts-typekit",