/** * External dependencies */ import { __ } from '@wordpress/i18n'; import { Button, TextControl, Notice, Spinner, CheckboxControl, } from '@wordpress/components'; import { FormInputValidation } from '@automattic/components'; import { SelectControl } from '@woocommerce/components'; import { Icon, chevronDown } from '@wordpress/icons'; import { createInterpolateElement, useEffect, useState, } from '@wordpress/element'; import { findCountryOption, getCountry } from '@woocommerce/onboarding'; import { decodeEntities } from '@wordpress/html-entities'; import { z } from 'zod'; import classNames from 'classnames'; /** * Internal dependencies */ import { CoreProfilerStateMachineContext, BusinessInfoEvent } from '../index'; import { CountryStateOption } from '../services/country'; import { Heading } from '../components/heading/heading'; import { Navigation } from '../components/navigation/navigation'; /** These are some store names that are known to be set by default and not likely to be used as actual names */ export const POSSIBLY_DEFAULT_STORE_NAMES = [ undefined, 'woocommerce', 'Site Title', '', ]; export type IndustryChoice = ( typeof industryChoices )[ number ][ 'key' ]; export const industryChoices = [ { label: __( 'Clothing and accessories', 'woocommerce' ), key: 'clothing_and_accessories' as const, }, { label: __( 'Health and beauty', 'woocommerce' ), key: 'health_and_beauty' as const, }, { label: __( 'Food and drink', 'woocommerce' ), key: 'food_and_drink' as const, }, { label: __( 'Home, furniture and garden', 'woocommerce' ), key: 'home_furniture_and_garden' as const, }, { label: __( 'Education and learning', 'woocommerce' ), key: 'education_and_learning' as const, }, { label: __( 'Electronics and computers', 'woocommerce' ), key: 'electronics_and_computers' as const, }, { label: __( 'Other', 'woocommerce' ), key: 'other' as const, }, ]; export type IndustryChoiceOption = ( typeof industryChoices )[ number ]; export const selectIndustryMapping = { im_just_starting_my_business: __( 'What type of products or services do you plan to sell?', 'woocommerce' ), im_already_selling: __( 'Which industry is your business in?', 'woocommerce' ), im_setting_up_a_store_for_a_client: __( "Which industry is your client's business in?", 'woocommerce' ), }; export type BusinessInfoContextProps = Pick< CoreProfilerStateMachineContext, 'geolocatedLocation' | 'userProfile' | 'businessInfo' | 'countries' > & { onboardingProfile: Pick< CoreProfilerStateMachineContext[ 'onboardingProfile' ], | 'industry' | 'business_choice' | 'is_store_country_set' | 'is_agree_marketing' | 'store_email' >; } & Partial< Pick< CoreProfilerStateMachineContext, 'currentUserEmail' > >; export const BusinessInfo = ( { context, navigationProgress, sendEvent, }: { context: BusinessInfoContextProps; navigationProgress: number; sendEvent: ( event: BusinessInfoEvent ) => void; } ) => { const { geolocatedLocation, userProfile: { businessChoice }, businessInfo, countries, onboardingProfile: { is_store_country_set: isStoreCountrySet, industry: industryFromOnboardingProfile, business_choice: businessChoiceFromOnboardingProfile, is_agree_marketing: isOptInMarketingFromOnboardingProfile, store_email: storeEmailAddressFromOnboardingProfile, }, currentUserEmail, } = context; const [ storeName, setStoreName ] = useState( businessInfo.storeName || '' ); const [ storeCountry, setStoreCountry ] = useState< CountryStateOption >( { key: '', label: '', } ); useEffect( () => { if ( isStoreCountrySet ) { const previouslyStoredCountryOption = countries.find( ( country ) => country.key === businessInfo.location ); setStoreCountry( previouslyStoredCountryOption || { key: '', label: '' } ); } }, [ businessInfo.location, countries, isStoreCountrySet ] ); const [ geolocationMatch, setGeolocationMatch ] = useState( { key: '', label: '', } ); useEffect( () => { if ( geolocatedLocation ) { const foundCountryOption = findCountryOption( countries, geolocatedLocation ); if ( foundCountryOption ) { setGeolocationMatch( foundCountryOption ); if ( ! isStoreCountrySet ) { setStoreCountry( foundCountryOption ); } } } }, [ countries, isStoreCountrySet, geolocatedLocation ] ); const geolocationOverruled = geolocatedLocation && getCountry( storeCountry.key ) !== getCountry( geolocationMatch.key ); const [ industry, setIndustry ] = useState< IndustryChoiceOption | undefined >( industryFromOnboardingProfile ? industryChoices.find( ( choice ) => choice.key === industryFromOnboardingProfile[ 0 ] ) : undefined ); const selectCountryLabel = __( 'Select country/region', 'woocommerce' ); const selectIndustryQuestionLabel = selectIndustryMapping[ businessChoice || businessChoiceFromOnboardingProfile || 'im_just_starting_my_business' ]; const [ dismissedGeolocationNotice, setDismissedGeolocationNotice ] = useState( false ); const [ hasSubmitted, setHasSubmitted ] = useState( false ); const [ isEmailInvalid, setIsEmailInvalid ] = useState( false ); const [ storeEmailAddress, setEmailAddress ] = useState( storeEmailAddressFromOnboardingProfile || currentUserEmail || '' ); const [ isOptInMarketing, setIsOptInMarketing ] = useState< boolean >( isOptInMarketingFromOnboardingProfile || false ); const [ doValidate, setDoValidate ] = useState( false ); useEffect( () => { if ( doValidate ) { const parseEmail = z .string() .email() .safeParse( storeEmailAddress ); setIsEmailInvalid( isOptInMarketing && ! parseEmail.success ); setDoValidate( false ); } }, [ isOptInMarketing, doValidate, storeEmailAddress ] ); return (
{ setStoreName( value ); } } value={ decodeEntities( storeName ) } label={ <> { __( 'Give your store a name', 'woocommerce' ) } } placeholder={ __( 'Ex. My awesome store', 'woocommerce' ) } />

{ __( "Don't worry — you can always change it later!", 'woocommerce' ) }

{ selectIndustryQuestionLabel }

} onChange={ ( results: Array< ( typeof industryChoices )[ number ] > ) => { if ( results.length ) { setIndustry( results[ 0 ] ); } } } selected={ industry ? [ industry ] : [] } showAllOnFocus isSearchable />

{ __( 'Where is your store located?', 'woocommerce' ) } { '*' }

{ return new RegExp( '(^' + query + '| — (' + query + '))', 'i' ); } } options={ countries } excludeSelectedOptions={ false } help={ } onChange={ ( results: Array< CountryStateOption > ) => { if ( results.length ) { setStoreCountry( results[ 0 ] ); } } } selected={ storeCountry ? [ storeCountry ] : [] } showAllOnFocus isSearchable /> { /* woocommerce-profiler-select-control__country-spacer exists purely because the select-control above has an unremovable and unstyleable div and that's preventing margin collapse */ }
{ geolocationOverruled && ! dismissedGeolocationNotice && ( setDismissedGeolocationNotice( true ) } status="warning" >

{ createInterpolateElement( __( // translators: first tag is filled with the country name detected by geolocation, second tag is the country name selected by the user "It looks like you're located in . Are you sure you want to create a store in ?", 'woocommerce' ), { geolocatedCountry: ( ), selectedCountry: ( { storeCountry.label } ), } ) }

{ __( 'Setting up your store in the wrong country may lead to the following issues: ', 'woocommerce' ) }

  • { __( 'Tax and duty obligations', 'woocommerce' ) }
  • { __( 'Payment issues', 'woocommerce' ) }
  • { __( 'Shipping issues', 'woocommerce' ) }
) } { <> { if ( isEmailInvalid ) { setDoValidate( true ); // trigger validation as we want to feedback to the user as soon as it becomes valid } setEmailAddress( value ); } } onBlur={ () => { setDoValidate( true ); } } value={ decodeEntities( storeEmailAddress ) } label={ <> { __( 'Your email address', 'woocommerce' ) } { isOptInMarketing && ( { '*' } ) } } placeholder={ __( 'wordpress@example.com', 'woocommerce' ) } /> { isEmailInvalid && ( ) } { setIsOptInMarketing( isChecked ); setDoValidate( true ); } } /> }
); };