/** * External dependencies */ import { __ } from '@wordpress/i18n'; import { Button, Card, CardBody, CardFooter, CheckboxControl, FlexItem as MaybeFlexItem, Spinner, Popover, } from '@wordpress/components'; import { Component, useRef } from '@wordpress/element'; import { compose } from '@wordpress/compose'; import { withDispatch, withSelect } from '@wordpress/data'; import { Form, TextControl } from '@woocommerce/components'; import { COUNTRIES_STORE_NAME, ONBOARDING_STORE_NAME, OPTIONS_STORE_NAME, SETTINGS_STORE_NAME, } from '@woocommerce/data'; import { recordEvent } from '@woocommerce/tracks'; import { Text } from '@woocommerce/experimental'; import { Icon, info } from '@wordpress/icons'; import { isEmail } from '@wordpress/url'; import { CurrencyContext } from '@woocommerce/currency'; /** * Internal dependencies */ import { getCountryCode, getCurrencyRegion } from '../../../dashboard/utils'; import { StoreAddress, getStoreAddressValidator, } from '../../../dashboard/components/settings/general/store-address'; import UsageModal from '../usage-modal'; import { getAdminSetting } from '~/utils/admin-settings'; import './style.scss'; // FlexItem is not available until WP version 5.5. This code is safe to remove // once the minimum WP supported version becomes 5.5. const FlextItemSubstitute = ( { children, align } ) => { const style = { display: 'flex', 'justify-content': align ? 'center' : 'flex-start', }; return
{ children }
; }; const FlexItem = MaybeFlexItem || FlextItemSubstitute; const LoadingPlaceholder = () => (
); export class StoreDetails extends Component { constructor( props ) { super( props ); this.state = { showUsageModal: false, skipping: false, isSkipSetupPopoverVisible: false, }; this.onContinue = this.onContinue.bind( this ); this.onSubmit = this.onSubmit.bind( this ); this.validateStoreDetails = this.validateStoreDetails.bind( this ); this.onFormValueChange = this.onFormValueChange.bind( this ); this.changedFormValues = {}; } componentDidUpdate() { if ( this.props.isLoading === false && Object.keys( this.changedFormValues ).length === 0 ) { // Make a copy of the initialValues. // The values in this object gets updated on onFormValueChange. this.changedFormValues = { ...this.props.initialValues }; this.props.trackStepValueChanges( this.props.step.key, this.props.initialValues, this.changedFormValues, () => { this.onContinue( this.changedFormValues ); } ); } } deriveCurrencySettings( countryState ) { if ( ! countryState ) { return null; } const Currency = this.context; const country = getCountryCode( countryState ); const { currencySymbols = {}, localeInfo = {} } = getAdminSetting( 'onboarding', {} ); return Currency.getDataForCountry( country, localeInfo, currencySymbols ); } onSubmit() { this.setState( { showUsageModal: true, skipping: false, } ); } onFormValueChange( changedFormValue ) { this.changedFormValues[ changedFormValue.name ] = changedFormValue.value; } async onContinue( values ) { const { createNotice, updateProfileItems, updateAndPersistSettingsForGroup, profileItems, settings, errorsRef, } = this.props; const currencySettings = this.deriveCurrencySettings( values.countryState ); const Currency = this.context; Currency.setCurrency( currencySettings ); recordEvent( 'storeprofiler_store_details_continue', { store_country: getCountryCode( values.countryState ), derived_currency: currencySettings.code, email_signup: values.isAgreeMarketing, } ); await updateAndPersistSettingsForGroup( 'general', { general: { ...settings, woocommerce_store_address: values.addressLine1, woocommerce_store_address_2: values.addressLine2, woocommerce_default_country: values.countryState, woocommerce_store_city: values.city, woocommerce_store_postcode: values.postCode, woocommerce_currency: currencySettings.code, woocommerce_currency_pos: currencySettings.symbolPosition, woocommerce_price_thousand_sep: currencySettings.thousandSeparator, woocommerce_price_decimal_sep: currencySettings.decimalSeparator, woocommerce_price_num_decimals: currencySettings.precision, }, } ); const profileItemsToUpdate = { is_agree_marketing: values.isAgreeMarketing, store_email: values.storeEmail, is_store_country_set: typeof values.countryState === 'string' && values.countryState !== '', }; const region = getCurrencyRegion( values.countryState ); /** * If a user has already selected cdb industry and returns to change to a * non US store, remove cbd industry. * * NOTE: the following call to `updateProfileItems` does not respect the * `await` and performs an update aysnchronously. This means the following * screen may not be initialized with correct profile settings. * * This comment may be removed when a refactor to wp.data datastores is complete. */ if ( region !== 'US' && profileItems.industry && profileItems.industry.length ) { const cbdSlug = 'cbd-other-hemp-derived-products'; const trimmedIndustries = profileItems.industry.filter( ( industry ) => { return cbdSlug !== industry && cbdSlug !== industry.slug; } ); profileItemsToUpdate.industry = trimmedIndustries; } let errorMessages = []; try { await updateProfileItems( profileItemsToUpdate ); } catch ( error ) { // Array of error messages obtained from API response. if ( error?.data?.params ) { errorMessages = Object.values( error.data.params ); } } if ( ! Boolean( errorsRef.current.settings ) && ! errorMessages.length ) { return true; } createNotice( 'error', __( 'There was a problem saving your store details', 'woocommerce' ) ); errorMessages.forEach( ( message ) => createNotice( 'error', message ) ); } validateStoreDetails( values ) { const { getLocale } = this.props; const locale = getLocale( values.countryState ); const validateAddress = getStoreAddressValidator( locale ); const errors = validateAddress( values ); if ( values.storeEmail && ! isEmail( values.storeEmail ) ) { errors.storeEmail = __( 'Invalid email address', 'woocommerce' ); } if ( values.isAgreeMarketing && ( ! values.storeEmail || ! values.storeEmail.trim().length ) ) { errors.storeEmail = __( 'Please enter your email address to subscribe', 'woocommerce' ); } return errors; } render() { const { showUsageModal, skipping, isSkipSetupPopoverVisible } = this.state; const { skipProfiler, isLoading, isBusy, initialValues, invalidateResolutionForStoreSelector, } = this.props; /* eslint-disable @wordpress/i18n-no-collapsible-whitespace */ const skipSetupText = __( 'Manual setup is only recommended for\n experienced WooCommerce users or developers.', 'woocommerce' ); /* eslint-enable @wordpress/i18n-no-collapsible-whitespace */ if ( isLoading ) { return (
); } return (
{ __( 'Welcome to WooCommerce', 'woocommerce' ) } { __( 'Tell us where you run your business to help us configure currency, shipping, taxes, and more in a fully automated way.', 'woocommerce' ) }
{ ( { getInputProps, handleSubmit, values, isValidForm, setValue, } ) => ( { showUsageModal && ( { if ( skipping ) { skipProfiler(); } else { this.onContinue( values ).then( () => this.props.goToNextStep() ); } } } onClose={ () => this.setState( { showUsageModal: false, skipping: false, } ) } /> ) }
{ __( 'Get tips, product updates and inspiration straight to your mailbox.', 'woocommerce' ) }{ ' ' } { __( 'Powered by Mailchimp', 'woocommerce' ) } } { ...getInputProps( 'isAgreeMarketing' ) } />
) }
{ isSkipSetupPopoverVisible && ( this.setState( { isSkipSetupPopoverVisible: false, } ) } > { skipSetupText } ) }
); } } StoreDetails.contextType = CurrencyContext; export default compose( withSelect( ( select ) => { const { getSettings, getSettingsError, isUpdateSettingsRequesting } = select( SETTINGS_STORE_NAME ); const { getProfileItems, isOnboardingRequesting, getEmailPrefill, hasFinishedResolution: hasFinishedResolutionOnboarding, } = select( ONBOARDING_STORE_NAME ); const { getLocale, getLocales, getCountries, hasFinishedResolution: hasFinishedResolutionCountries, } = select( COUNTRIES_STORE_NAME ); const { isResolving } = select( OPTIONS_STORE_NAME ); const profileItems = getProfileItems(); const emailPrefill = getEmailPrefill(); const { general: settings = {} } = getSettings( 'general' ); const isBusy = isOnboardingRequesting( 'updateProfileItems' ) || isUpdateSettingsRequesting( 'general' ) || isResolving( 'getOption', [ 'woocommerce_allow_tracking' ] ); const isLoading = ! hasFinishedResolutionOnboarding( 'getProfileItems' ) || ! hasFinishedResolutionOnboarding( 'getEmailPrefill' ) || ! hasFinishedResolutionCountries( 'getLocales' ) || ! hasFinishedResolutionCountries( 'getCountries' ); const errorsRef = useRef( { settings: null, } ); errorsRef.current = { settings: getSettingsError( 'general' ), }; // Check if a store country is set so that we don't default // to WooCommerce's default country of the US:CA. const countryState = profileItems.is_store_country_set ? settings.woocommerce_default_country : ''; getCountries(); getLocales(); const initialValues = { addressLine1: settings.woocommerce_store_address || '', addressLine2: settings.woocommerce_store_address_2 || '', city: settings.woocommerce_store_city || '', countryState, postCode: settings.woocommerce_store_postcode || '', // By default, the marketing checkbox should be unticked by default to comply with WordPress.org plugin review guidelines. isAgreeMarketing: typeof profileItems.is_agree_marketing === 'boolean' ? profileItems.is_agree_marketing : false, storeEmail: typeof profileItems.store_email === 'string' ? profileItems.store_email : emailPrefill, }; return { getLocale, initialValues, isLoading, profileItems, isBusy, settings, errorsRef, }; } ), withDispatch( ( dispatch ) => { const { createNotice } = dispatch( 'core/notices' ); const { invalidateResolutionForStoreSelector, updateProfileItems } = dispatch( ONBOARDING_STORE_NAME ); const { updateAndPersistSettingsForGroup } = dispatch( SETTINGS_STORE_NAME ); return { createNotice, invalidateResolutionForStoreSelector, updateProfileItems, updateAndPersistSettingsForGroup, }; } ) )( StoreDetails );