/** * External dependencies */ import { __ } from '@wordpress/i18n'; import { Component } from '@wordpress/element'; import { compose } from '@wordpress/compose'; import { Button, Card, CardBody, CardFooter, TabPanel, __experimentalText as Text, } from '@wordpress/components'; import { withDispatch, withSelect } from '@wordpress/data'; import { SelectControl, Form, TextControl } from '@woocommerce/components'; import { ONBOARDING_STORE_NAME, PLUGINS_STORE_NAME, SETTINGS_STORE_NAME, } from '@woocommerce/data'; import { recordEvent } from '@woocommerce/tracks'; /** * Internal dependencies */ import { CurrencyContext } from '~/lib/currency-context'; import { createNoticesFromResponse } from '~/lib/notices'; import { platformOptions } from '../../data/platform-options'; import { sellingVenueOptions } from '../../data/selling-venue-options'; import { getRevenueOptions } from '../../data/revenue-options'; import { getProductCountOptions } from '../../data/product-options'; import { SelectiveExtensionsBundle } from './selective-extensions-bundle'; import './style.scss'; const BUSINESS_DETAILS_TAB_NAME = 'business-details'; const FREE_FEATURES_TAB_NAME = 'free-features'; export const filterBusinessExtensions = ( extensionInstallationOptions ) => { return ( Object.keys( extensionInstallationOptions ) .filter( ( key ) => extensionInstallationOptions[ key ] && key !== 'install_extensions' ) .map( ( key ) => { // Remove anything after : // Please refer to selective-extensions-bundle/index.js // installableExtensions variable // this is to allow duplicate slugs (Tax & Shipping for example) return key.split( ':' )[ 0 ]; } ) // remove duplicate .filter( ( item, index, arr ) => arr.indexOf( item ) === index ) ); }; class BusinessDetails extends Component { constructor() { super(); this.state = { isPopoverVisible: false, isValid: false, currentTab: 'business-details', savedValues: null, }; this.onContinue = this.onContinue.bind( this ); this.validate = this.validate.bind( this ); } async onContinue( extensionInstallationOptions ) { const { createNotice, goToNextStep, installAndActivatePlugins, updateProfileItems, } = this.props; const { other_platform: otherPlatform, other_platform_name: otherPlatformName, product_count: productCount, revenue, selling_venues: sellingVenues, } = this.state.savedValues; const businessExtensions = filterBusinessExtensions( extensionInstallationOptions ); recordEvent( 'storeprofiler_store_business_features_continue', { all_extensions_installed: Object.values( extensionInstallationOptions ).every( ( val ) => val ), install_woocommerce_services: extensionInstallationOptions[ 'woocommerce-services:shipping' ] || extensionInstallationOptions[ 'woocommerce-services:tax' ], install_mailchimp: extensionInstallationOptions[ 'mailchimp-for-woocommerce' ], install_mailpoet: extensionInstallationOptions.mailpoet, install_jetpack: extensionInstallationOptions.jetpack, install_google_ads: extensionInstallationOptions[ 'google-listings-and-ads' ], install_facebook: extensionInstallationOptions[ 'facebook-for-woocommerce' ], install_wcpay: extensionInstallationOptions[ 'woocommerce-payments' ], install_creative_mail: extensionInstallationOptions[ 'creative-mail-by-constant-contact' ], } ); const updates = { other_platform: otherPlatform, other_platform_name: otherPlatform === 'other' ? otherPlatformName : '', product_count: productCount, revenue, selling_venues: sellingVenues, business_extensions: businessExtensions, }; // Remove possible empty values like `revenue` and `other_platform`. Object.keys( updates ).forEach( ( key ) => updates[ key ] === '' && delete updates[ key ] ); const promises = [ updateProfileItems( updates ).catch( () => { throw new Error(); } ), ]; if ( businessExtensions.length ) { promises.push( installAndActivatePlugins( businessExtensions ) .then( ( response ) => { createNoticesFromResponse( response ); } ) .catch( ( error ) => { createNoticesFromResponse( error ); throw new Error(); } ) ); } Promise.all( promises ) .then( () => { goToNextStep(); } ) .catch( () => { createNotice( 'error', __( 'There was a problem updating your business details', 'woocommerce-admin' ) ); } ); } validate( values ) { const errors = {}; if ( ! values.product_count.length ) { errors.product_count = __( 'This field is required', 'woocommerce-admin' ); } if ( ! values.selling_venues.length ) { errors.selling_venues = __( 'This field is required', 'woocommerce-admin' ); } if ( ! values.other_platform.length && [ 'other', 'brick-mortar-other' ].includes( values.selling_venues ) ) { errors.other_platform = __( 'This field is required', 'woocommerce-admin' ); } if ( ! values.other_platform_name && values.other_platform === 'other' && [ 'other', 'brick-mortar-other' ].includes( values.selling_venues ) ) { errors.other_platform_name = __( 'This field is required', 'woocommerce-admin' ); } if ( ! values.revenue.length && [ 'other', 'brick-mortar', 'brick-mortar-other', 'other-woocommerce', ].includes( values.selling_venues ) ) { errors.revenue = __( 'This field is required', 'woocommerce-admin' ); } if ( Object.keys( errors ).length === 0 ) { this.setState( { isValid: true } ); } return errors; } trackBusinessDetailsStep( { other_platform: otherPlatform, other_platform_name: otherPlatformName, product_count: productCount, selling_venues: sellingVenues, revenue, } ) { const { getCurrencyConfig } = this.context; recordEvent( 'storeprofiler_store_business_details_continue_variant', { already_selling: sellingVenues, currency: getCurrencyConfig().code, product_number: productCount, revenue, used_platform: otherPlatform, used_platform_name: otherPlatformName, } ); } renderBusinessDetailsStep() { const { goToNextStep, isInstallingActivating, hasInstallActivateError, } = this.props; const { formatAmount, getCurrencyConfig } = this.context; const productCountOptions = getProductCountOptions( getCurrencyConfig() ); return (
{ this.setState( { savedValues: values, currentTab: 'free-features', } ); this.trackBusinessDetailsStep( values ); } } onChange={ ( _, values, isValid ) => { this.setState( { savedValues: values, isValid } ); } } validate={ this.validate } > { ( { getInputProps, handleSubmit, values, isValidForm } ) => { return ( <>
{ __( 'Tell us about your business', 'woocommerce-admin' ) } { __( "We'd love to know if you are just getting started or you already have a business in place.", 'woocommerce-admin' ) }
{ [ 'other', 'brick-mortar', 'brick-mortar-other', 'other-woocommerce', ].includes( values.selling_venues ) && ( ) } { [ 'other', 'brick-mortar-other', ].includes( values.selling_venues ) && ( <>
{ values.other_platform === 'other' && ( ) }
) }
{ hasInstallActivateError && ( ) }
); } }
); } renderFreeFeaturesStep() { const { isInstallingActivating, settings, profileItems } = this.props; const country = settings.woocommerce_default_country ? settings.woocommerce_default_country : null; return ( <>
{ __( 'Included business features', 'woocommerce-admin' ) } { __( 'We recommend enhancing your store with these free extensions', 'woocommerce-admin' ) } { __( 'No commitment required - you can remove them at any time.', 'woocommerce-admin' ) }
); } render() { const { initialValues } = this.props; // There is a hack here to help us manage the selected tab programatically. // We set the tab name "current-tab". when its the one we want selected. This tricks // the logic in the TabPanel and allows us to switch which tab has the name "current-tab" // and force it to re-render with a different tab selected. return ( { if ( this.state.currentTab !== tabName ) { this.setState( { currentTab: tabName, savedValues: this.state.savedValues || initialValues, } ); } } } tabs={ [ { name: this.state.currentTab === BUSINESS_DETAILS_TAB_NAME ? 'current-tab' : BUSINESS_DETAILS_TAB_NAME, id: BUSINESS_DETAILS_TAB_NAME, title: __( 'Business details', 'woocommerce-admin' ), }, { name: this.state.currentTab === FREE_FEATURES_TAB_NAME ? 'current-tab' : FREE_FEATURES_TAB_NAME, id: FREE_FEATURES_TAB_NAME, title: __( 'Free features', 'woocommerce-admin' ), className: this.state.isValid ? '' : 'is-disabled', }, ] } > { ( tab ) => <>{ this.getTab( tab.id ) } } ); } getTab( tabId ) { if ( tabId === BUSINESS_DETAILS_TAB_NAME ) { return this.renderBusinessDetailsStep(); } return this.renderFreeFeaturesStep(); } } BusinessDetails.contextType = CurrencyContext; export const BusinessFeaturesList = compose( withSelect( ( select ) => { const { getSettings, getSettingsError } = select( SETTINGS_STORE_NAME ); const { getProfileItems, getOnboardingError } = select( ONBOARDING_STORE_NAME ); const { getPluginsError, isPluginsRequesting } = select( PLUGINS_STORE_NAME ); const { general: settings = {} } = getSettings( 'general' ); return { hasInstallActivateError: getPluginsError( 'installPlugins' ) || getPluginsError( 'activatePlugins' ), isError: Boolean( getOnboardingError( 'updateProfileItems' ) ), profileItems: getProfileItems(), isSettingsError: Boolean( getSettingsError( 'general' ) ), settings, isInstallingActivating: isPluginsRequesting( 'installPlugins' ) || isPluginsRequesting( 'activatePlugins' ) || isPluginsRequesting( 'getJetpackConnectUrl' ), }; } ), withDispatch( ( dispatch ) => { const { updateProfileItems } = dispatch( ONBOARDING_STORE_NAME ); const { installAndActivatePlugins } = dispatch( PLUGINS_STORE_NAME ); const { createNotice } = dispatch( 'core/notices' ); return { createNotice, installAndActivatePlugins, updateProfileItems, }; } ) )( BusinessDetails );