.dq7btjj > div > div{opacity:1 !important;}
/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["src/screens/ClaimSubdomain/steps/ChangeStep/ChangeStep.tsx"],"names":[".dq7btjj"],"mappings":"AA0QMA","file":"src/screens/ClaimSubdomain/steps/ChangeStep/ChangeStep.tsx","sourcesContent":["import { memo, useCallback, useMemo, useState } from 'react';\nimport { defineMessages, useIntl } from 'react-intl';\nimport { useHistory } from 'react-router-dom';\nimport { styled } from '@linaria/react';\nimport {\n  logSubdomainClaim,\n  logSubdomainClaimError,\n  logSubdomainClaimSuccess,\n  useLogSubdomainChangeStepViewed,\n} from 'cb-wallet-analytics/subdomain';\nimport { cbReportError } from 'cb-wallet-data/errors/reportError';\nimport { useDebouncedCallback } from 'cb-wallet-data/hooks/useDebouncedCallback';\nimport { useValidateUsername } from 'cb-wallet-data/stores/User/hooks/useValidateUsername';\nimport { PERF_MARKS, SUBDOMAIN_MARKS } from 'cb-wallet-data/utils/perf-constants';\nimport { ValidatedTextInput } from 'wallet-cds-extension/components/ValidatedTextInput/ValidatedTextInput';\nimport { useStepper } from 'wallet-cds-extension/external-libs/stepper';\nimport { useExtensionPerfMark } from 'wallet-cds-extension/hooks/useExtensionPerfMark/useExtensionPerfMark';\nimport { useSubdomainReservation } from 'wallet-cds-extension/hooks/useSubdomainReservation';\nimport { useToast } from 'wallet-cds-extension/overrides/@cbhq/cds-web/overlays/useToast';\nimport { useToggler } from '@cbhq/cds-web';\nimport { Button } from '@cbhq/cds-web/buttons';\nimport { TextInput } from '@cbhq/cds-web/controls';\nimport { HStack, VStack } from '@cbhq/cds-web/layout';\nimport { Spinner } from '@cbhq/cds-web/loaders/Spinner';\nimport { TextBody, TextTitle1 } from '@cbhq/cds-web/typography';\n\nimport { usePerfMarkOnMount } from ':extension/hooks/usePerfMarkOnMount/usePerfMarkOnMount';\nimport { StepWrapper } from ':extension/screens/ClaimSubdomain/components/StepWrapper';\nimport { TermsCheckbox } from ':extension/screens/ClaimSubdomain/components/TermsCheckbox';\nimport { stepper } from ':extension/screens/ClaimSubdomain/flow';\nimport {\n  messages as validationMessages,\n  useUsernameValidators,\n} from ':extension/screens/Onboarding/hooks/useUsernameValidators';\nimport { RoutesEnum } from ':extension/screens/Routes/Routes';\n\nexport const messages = defineMessages({\n  title: {\n    defaultMessage: 'Change username',\n    description: 'Page title to change the username for the subdomain',\n  },\n  subtitle: {\n    defaultMessage: 'You can only change your username once a year.',\n    description: 'Short description about limitation of username change frequency',\n  },\n  checking: {\n    defaultMessage: 'Checking availability…',\n    description: 'Text to display when we are checking if the username exists',\n  },\n  validationAvailable: {\n    defaultMessage: 'Available',\n    description: 'Validation message saying the username is not already taken by another user',\n  },\n  currentLabel: {\n    defaultMessage: 'Current name',\n    description: 'the name the user currently has for their account or subdomain',\n  },\n  newLabel: {\n    defaultMessage: 'New name',\n    description: 'the name the user wants to use for their account or subdomain',\n  },\n  error: {\n    defaultMessage: 'Unable to claim subdomain',\n    description: 'Error message for when claiming a subdomain fails',\n  },\n  submit: {\n    defaultMessage: 'Submit',\n    description: 'Label for button CTA to move to next step in claim flow',\n  },\n  back: {\n    defaultMessage: 'Go back',\n    description: 'Label for button CTA to go back to the reservation step',\n  },\n  poweredBy: {\n    defaultMessage: 'Powered by ENS',\n    description: 'Label explaining that the new usernames are powered by ENS',\n  },\n});\n\nconst DISALLOW_ANONYMOUS = false;\nconst IS_SUBDOMAIN = true;\n\nexport const ChangeStep = memo(function ChangeStep() {\n  const { isLoading, username, subdomainSuffix, claim } = useSubdomainReservation();\n  const validateUsername = useValidateUsername();\n  const validators = useUsernameValidators(DISALLOW_ANONYMOUS, IS_SUBDOMAIN);\n  const { goToStep } = useStepper(stepper);\n  const [isTermsChecked, { toggle: handleTermsChange }] = useToggler();\n  const [isUsernameAvailable, setIsUsernameAvailable] = useState(false);\n  const [newUsername, setNewUsername] = useState('');\n  const [isValidating, setIsValidating] = useState(false);\n  const [isClaiming, setIsClaiming] = useState(false);\n  const [hasErrors, setHasErrors] = useState(false);\n  const { push } = useHistory();\n  const { formatMessage } = useIntl();\n  const toast = useToast();\n\n  const { start: markStartClaimStepPerf } = useExtensionPerfMark(\n    PERF_MARKS.bridge,\n    SUBDOMAIN_MARKS.load_subdomain_claim,\n    {\n      logMetric: true,\n    },\n  );\n\n  const givenError = useMemo(() => {\n    // only default to \"Not Available\" if all validators come back clean\n    if (!hasErrors && !isUsernameAvailable) {\n      return formatMessage(validationMessages.validationNotAvailable);\n    }\n  }, [formatMessage, hasErrors, isUsernameAvailable]);\n\n  const isDisabled = useMemo(\n    () => !isTermsChecked || hasErrors || !newUsername || isValidating || !isUsernameAvailable,\n    [hasErrors, isValidating, isTermsChecked, newUsername, isUsernameAvailable],\n  );\n\n  const handleBackPress = useCallback(() => {\n    markStartClaimStepPerf();\n    goToStep('claim');\n  }, [goToStep, markStartClaimStepPerf]);\n\n  const handleClaim = useCallback(async () => {\n    if (isDisabled) {\n      return;\n    }\n\n    logSubdomainClaim();\n    setIsClaiming(true);\n\n    try {\n      await claim(newUsername);\n      setIsClaiming(false);\n      logSubdomainClaimSuccess();\n      push(RoutesEnum.DEFAULT);\n    } catch (error) {\n      logSubdomainClaimError();\n      cbReportError({\n        error: error as Error,\n        context: 'did_error',\n        metadata: {\n          screen: 'ChangeStep',\n          method: 'ChangeStep.handleClaim',\n        },\n      });\n      setIsClaiming(false);\n      toast.show(formatMessage(messages.error));\n    }\n  }, [push, toast, formatMessage, isDisabled, claim, newUsername]);\n\n  const validateName = useCallback(\n    async (name: string) => {\n      try {\n        // If the user enters their current username, skip checking if the username is available\n        // since they already exist and would fail the check.\n        const isClaimingSameName = name.toLowerCase() === username?.toLowerCase();\n        setIsUsernameAvailable(isClaimingSameName || (await validateUsername(name, IS_SUBDOMAIN)));\n      } finally {\n        setIsValidating(false);\n      }\n    },\n    [validateUsername, username],\n  );\n\n  const debouncedValidateUsername = useDebouncedCallback(validateName);\n\n  const handleUsernameChange = useCallback(\n    (name: string): void => {\n      setIsValidating(true);\n      setNewUsername(name);\n      debouncedValidateUsername(name);\n    },\n    [debouncedValidateUsername],\n  );\n\n  useLogSubdomainChangeStepViewed();\n\n  usePerfMarkOnMount(PERF_MARKS.subdomain, SUBDOMAIN_MARKS.load_subdomain_change, {\n    type: 'end',\n  });\n\n  if (isLoading) {\n    return (\n      <StepWrapper testID=\"claim-subdomain-change-step\">\n        <VStack spacingHorizontal={3}>\n          <TextTitle1 as=\"h1\" spacingBottom={0.5}>\n            {formatMessage(messages.title)}\n          </TextTitle1>\n          <TextBody as=\"p\" color=\"foregroundMuted\">\n            {formatMessage(messages.subtitle)}\n          </TextBody>\n        </VStack>\n        <VStack alignItems=\"center\" justifyContent=\"center\" flexGrow={1}>\n          <Spinner testID=\"change-step-loading-spinner\" size={4} color=\"primary\" />\n        </VStack>\n      </StepWrapper>\n    );\n  }\n\n  return (\n    <StepWrapper testID=\"claim-subdomain-change-step\">\n      <VStack spacingHorizontal={3}>\n        <TextTitle1 as=\"h1\" spacingBottom={0.5}>\n          {formatMessage(messages.title)}\n        </TextTitle1>\n        <TextBody as=\"p\" color=\"foregroundMuted\">\n          {formatMessage(messages.subtitle)}\n        </TextBody>\n      </VStack>\n\n      <VStack spacingHorizontal={3} spacingTop={3} gap={1} flexGrow={1}>\n        <DisabledTransparency>\n          <TextInput\n            name=\"current-username\"\n            disabled\n            variant=\"foregroundMuted\"\n            label={formatMessage(messages.currentLabel)}\n            placeholder={username}\n            suffix={subdomainSuffix}\n          />\n        </DisabledTransparency>\n        <ValidatedTextInput\n          name=\"username\"\n          testID=\"username-input\"\n          defaultSubscript={formatMessage(messages.poweredBy)}\n          isLoading={isValidating}\n          loadingMessage={formatMessage(messages.checking)}\n          successMessage={formatMessage(messages.validationAvailable)}\n          label={formatMessage(messages.newLabel)}\n          value={newUsername}\n          onError={setHasErrors}\n          autoFocus\n          givenError={givenError}\n          onChange={handleUsernameChange}\n          validators={validators}\n          onEnter={handleClaim}\n          suffix={subdomainSuffix}\n        />\n      </VStack>\n\n      <VStack spacingHorizontal={3} spacingBottom={2} gap={1}>\n        <TermsCheckbox\n          testID=\"terms-checkbox\"\n          checked={isTermsChecked}\n          onChange={handleTermsChange}\n        />\n      </VStack>\n\n      <HStack spacingHorizontal={3} spacingBottom={2} gap={1}>\n        <Button block variant=\"secondary\" onPress={handleBackPress}>\n          {formatMessage(messages.back)}\n        </Button>\n        <Button\n          block\n          testID=\"submit-username-button\"\n          loading={isClaiming}\n          disabled={isDisabled}\n          onPress={handleClaim}\n        >\n          {formatMessage(messages.submit)}\n        </Button>\n      </HStack>\n    </StepWrapper>\n  );\n});\n\nconst DisabledTransparency = styled.div`\n  & > div > div {\n    opacity: 1 !important;\n  }\n`;\n"]}*/
.sscjc7e{position:-webkit-sticky;position:sticky;background-color:var(--background);bottom:0;z-index:999;}
/*# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["src/screens/ClaimSubdomain/steps/ClaimStep/ClaimStep.tsx"],"names":[".sscjc7e"],"mappings":"AA4MMA","file":"src/screens/ClaimSubdomain/steps/ClaimStep/ClaimStep.tsx","sourcesContent":["import { memo, ReactNode, useCallback, useState } from 'react';\nimport { defineMessages, useIntl } from 'react-intl';\nimport { useHistory } from 'react-router-dom';\nimport { styled } from '@linaria/react';\nimport {\n  logENSLearnHowLinkClick,\n  logSubdomainClaim,\n  logSubdomainClaimError,\n  logSubdomainClaimSuccess,\n  useLogSubdomainClaimStepViewed,\n} from 'cb-wallet-analytics/subdomain';\nimport { cbReportError } from 'cb-wallet-data/errors/reportError';\nimport { PERF_MARKS, SUBDOMAIN_MARKS } from 'cb-wallet-data/utils/perf-constants';\nimport { useStepper } from 'wallet-cds-extension/external-libs/stepper';\nimport { useExtensionPerfMark } from 'wallet-cds-extension/hooks/useExtensionPerfMark/useExtensionPerfMark';\nimport { useSubdomainReservation } from 'wallet-cds-extension/hooks/useSubdomainReservation';\nimport { useToast } from 'wallet-cds-extension/overrides/@cbhq/cds-web/overlays/useToast';\nimport { useToggler } from '@cbhq/cds-web';\nimport { Button } from '@cbhq/cds-web/buttons';\nimport { SpotRectangle } from '@cbhq/cds-web/illustrations';\nimport { Box, HStack, VStack } from '@cbhq/cds-web/layout';\nimport { Spinner } from '@cbhq/cds-web/loaders/Spinner';\nimport { Link, TextLabel2, TextTitle1, TextTitle3, TextTitle4 } from '@cbhq/cds-web/typography';\n\nimport { usePerfMarkOnMount } from ':extension/hooks/usePerfMarkOnMount/usePerfMarkOnMount';\nimport { StepWrapper } from ':extension/screens/ClaimSubdomain/components/StepWrapper';\nimport { TermsCheckbox } from ':extension/screens/ClaimSubdomain/components/TermsCheckbox';\nimport { stepper } from ':extension/screens/ClaimSubdomain/flow';\nimport { RoutesEnum } from ':extension/screens/Routes/Routes';\nimport { getLinkENSLearnHow } from ':extension/util/constants/externalLinks';\n\nexport const messages = defineMessages({\n  title: {\n    defaultMessage: 'We’re updating all usernames',\n    description: 'Page title explaining that this username has been reserved',\n  },\n  subtitle: {\n    defaultMessage:\n      'Coinbase Wallet usernames are becoming decentralized, powered by ENS. <linkUseENS>Learn more</linkUseENS>.',\n    description:\n      'Short description that the user can switch their username or use their ENS domain',\n  },\n  error: {\n    defaultMessage: 'Unable to claim subdomain',\n    description: 'Error message for when claiming a subdomain fails',\n  },\n  claim: {\n    defaultMessage: 'Accept',\n    description: 'Label for button CTA to move to next step in claim flow',\n  },\n  decline: {\n    defaultMessage: 'Decline',\n    description: 'Label for button CTA to exit claim flow',\n  },\n  change: {\n    defaultMessage: 'Change username',\n    description: 'Label for button CTA to change username',\n  },\n  poweredBy: {\n    defaultMessage: 'Powered by ENS',\n    description: 'Label explaining that the new usernames are powered by ENS',\n  },\n  back: {\n    defaultMessage: 'Back',\n    description: 'Label for button CTA to back out of the claim flow',\n  },\n});\n\nexport const ClaimStep = memo(function ClaimStep() {\n  const { isLoading, username, subdomainSuffix, claim } = useSubdomainReservation();\n  const { goToStep } = useStepper(stepper);\n  const [isTermsChecked, { toggle: handleTermsChange }] = useToggler();\n  const [isClaiming, setIsClaiming] = useState(false);\n  const { push, goBack } = useHistory();\n  const { formatMessage, locale } = useIntl();\n  const toast = useToast();\n\n  const { start: markStartRevokeStepPerf } = useExtensionPerfMark(\n    PERF_MARKS.bridge,\n    SUBDOMAIN_MARKS.load_subdomain_revoke,\n    {\n      logMetric: true,\n    },\n  );\n\n  const handleLinkUseENSPress = useCallback(() => {\n    logENSLearnHowLinkClick();\n    chrome.tabs.create({\n      url: getLinkENSLearnHow(locale),\n      active: false,\n    });\n  }, [locale]);\n\n  const linkUseENS = useCallback(\n    (text: ReactNode[]) => (\n      <Link variant=\"label2\" onPress={handleLinkUseENSPress}>\n        {text}\n      </Link>\n    ),\n    [handleLinkUseENSPress],\n  );\n\n  const handleDeclinePress = useCallback(() => {\n    markStartRevokeStepPerf();\n    goToStep('revoke');\n  }, [goToStep, markStartRevokeStepPerf]);\n\n  const handleClaimPress = useCallback(async () => {\n    if (!isTermsChecked) {\n      return;\n    }\n\n    logSubdomainClaim();\n    setIsClaiming(true);\n\n    try {\n      await claim(username!);\n      setIsClaiming(false);\n      logSubdomainClaimSuccess();\n      push(RoutesEnum.DEFAULT);\n    } catch (err) {\n      logSubdomainClaimError();\n      cbReportError({\n        error: err as Error,\n        context: 'did_error',\n        metadata: {\n          screen: 'ClaimStep',\n          method: 'ClaimStep.handleClaimPress',\n        },\n      });\n      setIsClaiming(false);\n      toast.show(formatMessage(messages.error));\n    }\n  }, [isTermsChecked, formatMessage, claim, username, push, toast]);\n\n  useLogSubdomainClaimStepViewed();\n\n  usePerfMarkOnMount(PERF_MARKS.subdomain, SUBDOMAIN_MARKS.load_subdomain_claim, {\n    type: 'end',\n  });\n\n  if (isLoading) {\n    return (\n      <StepWrapper testID=\"claim-subdomain-claim-step\" justifyContent=\"flex-start\">\n        <VStack spacingHorizontal={3}>\n          <TextTitle1 as=\"h1\" spacingBottom={0.5}>\n            {formatMessage(messages.title)}\n          </TextTitle1>\n          <TextLabel2 as=\"p\" color=\"foregroundMuted\">\n            {formatMessage(messages.subtitle, { linkUseENS })}\n          </TextLabel2>\n        </VStack>\n        <VStack alignItems=\"center\" justifyContent=\"center\" flexGrow={1}>\n          <Spinner size={4} color=\"primary\" />\n        </VStack>\n        <Box spacingHorizontal={3} spacingTop={0} spacingBottom={2}>\n          <Button variant=\"secondary\" block onPress={goBack}>\n            {formatMessage(messages.back)}\n          </Button>\n        </Box>\n      </StepWrapper>\n    );\n  }\n\n  return (\n    <StepWrapper testID=\"claim-subdomain-claim-step\" overflow=\"scroll\">\n      <VStack spacingHorizontal={3}>\n        <TextTitle1 as=\"h1\" spacingBottom={1} align=\"center\">\n          {formatMessage(messages.title)}\n        </TextTitle1>\n        <TextLabel2 as=\"p\" color=\"foregroundMuted\" align=\"center\">\n          {formatMessage(messages.subtitle, { linkUseENS })}\n        </TextLabel2>\n\n        <Box justifyContent=\"center\" spacingTop={3} spacingBottom={2}>\n          <SpotRectangle name=\"ethAddress\" dimension=\"240x120\" />\n        </Box>\n\n        <HStack justifyContent=\"center\">\n          <TextTitle3 as=\"span\">{username}</TextTitle3>\n          <TextTitle4 as=\"span\" color=\"foregroundMuted\">\n            {subdomainSuffix}\n          </TextTitle4>\n        </HStack>\n      </VStack>\n\n      <VStack spacingHorizontal={3} gap={1} spacingVertical={2}>\n        <TermsCheckbox checked={isTermsChecked} onChange={handleTermsChange} />\n      </VStack>\n      <StickyContainer>\n        <VStack spacingHorizontal={3} spacingTop={1} spacingBottom={0} gap={1}>\n          <Button block loading={isClaiming} disabled={!isTermsChecked} onPress={handleClaimPress}>\n            {formatMessage(messages.claim)}\n          </Button>\n\n          <Button variant=\"secondary\" block onPress={handleDeclinePress}>\n            {formatMessage(messages.decline)}\n          </Button>\n        </VStack>\n      </StickyContainer>\n    </StepWrapper>\n  );\n});\n\nconst StickyContainer = styled.div`\n  position: sticky;\n  background-color: var(--background);\n  bottom: 0;\n  z-index: 999;\n`;\n"]}*/

/*# sourceMappingURL=7927.css.map*/