setBudgetAndReserves.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. import {Component} from 'react';
  2. import styled from '@emotion/styled';
  3. import {Button} from 'sentry/components/button';
  4. import {Tag} from 'sentry/components/core/badge/tag';
  5. import RangeSlider from 'sentry/components/forms/controls/rangeSlider';
  6. import Panel from 'sentry/components/panels/panel';
  7. import PanelBody from 'sentry/components/panels/panelBody';
  8. import PanelFooter from 'sentry/components/panels/panelFooter';
  9. import PanelItem from 'sentry/components/panels/panelItem';
  10. import QuestionTooltip from 'sentry/components/questionTooltip';
  11. import {DATA_CATEGORY_INFO} from 'sentry/constants';
  12. import {t, tct} from 'sentry/locale';
  13. import {space} from 'sentry/styles/space';
  14. import {DataCategoryExact} from 'sentry/types/core';
  15. import {OnDemandBudgetMode, type OnDemandBudgets} from 'getsentry/types';
  16. import {
  17. formatReservedWithUnits,
  18. isBizPlanFamily,
  19. isDeveloperPlan,
  20. } from 'getsentry/utils/billing';
  21. import {getPlanCategoryName} from 'getsentry/utils/dataCategory';
  22. import trackGetsentryAnalytics from 'getsentry/utils/trackGetsentryAnalytics';
  23. import StepHeader from 'getsentry/views/amCheckout/steps/stepHeader';
  24. import {getDataCategoryTooltipText} from 'getsentry/views/amCheckout/steps/utils';
  25. import type {StepProps} from 'getsentry/views/amCheckout/types';
  26. import * as utils from 'getsentry/views/amCheckout/utils';
  27. import PayAsYouGoBudgetEdit from 'getsentry/views/onDemandBudgets/payAsYouGoBudgetEdit';
  28. import {getTotalBudget} from 'getsentry/views/onDemandBudgets/utils';
  29. const ATTACHMENT_DIGITS = 2;
  30. const PAYG_BUSINESS_DEFAULT = 30000;
  31. const PAYG_TEAM_DEFAULT = 10000;
  32. type Props = StepProps;
  33. type State = {
  34. // Once the PAYG budget is updated, we no longer suggest a new default PAYG value
  35. isUpdated: boolean;
  36. };
  37. class SetBudgetAndReserves extends Component<Props, State> {
  38. state: State = {isUpdated: false};
  39. componentDidUpdate(prevProps: Props) {
  40. const {isActive, organization, subscription, activePlan} = this.props;
  41. // record when step is opened
  42. if (prevProps.isActive || !isActive) {
  43. return;
  44. }
  45. const hasPartnerMigrationFeature = organization?.features.includes(
  46. 'partner-billing-migration'
  47. );
  48. // set default budget for new customers for the first time they complete plan selection
  49. if (
  50. (isDeveloperPlan(subscription.planDetails) || hasPartnerMigrationFeature) &&
  51. !this.state.isUpdated &&
  52. isActive
  53. ) {
  54. // Default shared budgets are hardcoded vs being a multiple of the plan's base price
  55. const defaultBudget = isBizPlanFamily(activePlan)
  56. ? PAYG_BUSINESS_DEFAULT
  57. : PAYG_TEAM_DEFAULT;
  58. this.handleBudgetChange({
  59. budgetMode: OnDemandBudgetMode.SHARED,
  60. sharedMaxBudget: defaultBudget,
  61. });
  62. this.setState({isUpdated: true});
  63. }
  64. if (organization) {
  65. trackGetsentryAnalytics('checkout.data_sliders_viewed', {
  66. organization,
  67. });
  68. }
  69. }
  70. get title() {
  71. return t('Set Your Pay-as-you-go Budget');
  72. }
  73. handleBudgetChange(value: OnDemandBudgets) {
  74. const {organization, subscription, onUpdate, formData} = this.props;
  75. // right now value is always a SharedOnDemandBudget but re-defining it here makes TS happy
  76. const budget = {
  77. budgetMode: value.budgetMode,
  78. sharedMaxBudget: getTotalBudget(value),
  79. };
  80. onUpdate({
  81. onDemandBudget: value,
  82. onDemandMaxSpend: budget.sharedMaxBudget,
  83. });
  84. if (organization) {
  85. trackGetsentryAnalytics('checkout.payg_changed', {
  86. organization,
  87. subscription,
  88. plan: formData.plan,
  89. cents: budget.sharedMaxBudget || 0,
  90. });
  91. }
  92. }
  93. handleReservedChange(value: number, category: string) {
  94. const {organization, onUpdate, formData} = this.props;
  95. onUpdate({reserved: {...formData.reserved, [category]: value}});
  96. if (organization) {
  97. trackGetsentryAnalytics('checkout.data_slider_changed', {
  98. organization,
  99. data_type: category,
  100. quantity: value,
  101. });
  102. }
  103. }
  104. renderBody = () => {
  105. const {formData, activePlan, checkoutTier} = this.props;
  106. const budgetIsNotUnset =
  107. typeof formData.onDemandMaxSpend === 'number' && !isNaN(formData.onDemandMaxSpend);
  108. const paygBudget: OnDemandBudgets =
  109. budgetIsNotUnset && formData.onDemandBudget
  110. ? formData.onDemandBudget.budgetMode === OnDemandBudgetMode.PER_CATEGORY
  111. ? {
  112. budgetMode: OnDemandBudgetMode.SHARED,
  113. sharedMaxBudget: getTotalBudget(formData.onDemandBudget),
  114. }
  115. : formData.onDemandBudget
  116. : {budgetMode: OnDemandBudgetMode.SHARED, sharedMaxBudget: 0};
  117. return (
  118. <PanelBody data-test-id={this.title}>
  119. <PayAsYouGoBudgetEdit
  120. payAsYouGoBudget={paygBudget}
  121. setPayAsYouGoBudget={value => this.handleBudgetChange(value)}
  122. />
  123. <RowWithTag>
  124. <SectionHeader>
  125. <LargeTitle>
  126. {t('Set Reserved Volumes')}
  127. <OptionalText>{t(' (optional)')}</OptionalText>
  128. <QuestionTooltip
  129. title={t('Prepay for usage by reserving volumes and save up to 20%')}
  130. position="bottom"
  131. size="sm"
  132. />
  133. </LargeTitle>
  134. </SectionHeader>
  135. <Tag type="promotion">{t('Plan ahead and save 20%')}</Tag>
  136. </RowWithTag>
  137. {activePlan.checkoutCategories
  138. .filter(
  139. // only show sliders for checkout categories with more than 1 bucket
  140. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  141. category => activePlan.planCategories[category].length > 1
  142. )
  143. .map(category => {
  144. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  145. const allowedValues = activePlan.planCategories[category].map(
  146. (bucket: any) => bucket.events
  147. );
  148. const eventBucket = utils.getBucket({
  149. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  150. events: formData.reserved[category],
  151. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  152. buckets: activePlan.planCategories[category],
  153. });
  154. const min = allowedValues[0];
  155. const max = allowedValues.slice(-1)[0];
  156. const billingInterval = utils.getShortInterval(activePlan.billingInterval);
  157. const price = utils.displayPrice({cents: eventBucket.price});
  158. const unitPrice = utils.displayUnitPrice({
  159. cents: eventBucket.unitPrice || 0,
  160. ...(category === DATA_CATEGORY_INFO[DataCategoryExact.ATTACHMENT].plural
  161. ? {
  162. minDigits: ATTACHMENT_DIGITS,
  163. maxDigits: ATTACHMENT_DIGITS,
  164. }
  165. : {}),
  166. });
  167. const sliderId = `slider-${category}`;
  168. return (
  169. <DataVolumeItem key={category} data-test-id={`${category}-volume-item`}>
  170. <div>
  171. <SectionHeader>
  172. <Title htmlFor={sliderId}>
  173. <div>{getPlanCategoryName({plan: activePlan, category})}</div>
  174. <QuestionTooltip
  175. title={getDataCategoryTooltipText(checkoutTier, category)}
  176. position="top"
  177. size="xs"
  178. />
  179. </Title>
  180. <Events>
  181. {
  182. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  183. formatReservedWithUnits(formData.reserved[category], category)
  184. }
  185. </Events>
  186. </SectionHeader>
  187. <Description>
  188. <div>
  189. {eventBucket.price !== 0 &&
  190. tct('[unitPrice] per [category]', {
  191. category:
  192. category ===
  193. DATA_CATEGORY_INFO[DataCategoryExact.ATTACHMENT].plural
  194. ? 'GB'
  195. : category ===
  196. DATA_CATEGORY_INFO[DataCategoryExact.SPAN].plural
  197. ? 'unit'
  198. : 'event',
  199. unitPrice,
  200. })}
  201. </div>
  202. <div>
  203. {eventBucket.price === 0
  204. ? t('included')
  205. : `${price}/${billingInterval}`}
  206. </div>
  207. </Description>
  208. </div>
  209. <div>
  210. <RangeSlider
  211. showLabel={false}
  212. name={category}
  213. id={sliderId}
  214. // @ts-expect-error TS(7053): Element implicitly has an 'any' type because expre... Remove this comment to see the full error message
  215. value={formData.reserved[category]}
  216. allowedValues={allowedValues}
  217. formatLabel={() => null}
  218. onChange={value =>
  219. value && this.handleReservedChange(value, category)
  220. }
  221. />
  222. <MinMax>
  223. <div>{utils.getEventsWithUnit(min, category)}</div>
  224. <div>{utils.getEventsWithUnit(max, category)}</div>
  225. </MinMax>
  226. </div>
  227. </DataVolumeItem>
  228. );
  229. })}
  230. </PanelBody>
  231. );
  232. };
  233. renderFooter = () => {
  234. const {stepNumber, onCompleteStep} = this.props;
  235. return (
  236. <StepFooter data-test-id={this.title}>
  237. <Button priority="primary" onClick={() => onCompleteStep(stepNumber)}>
  238. {t('Continue')}
  239. </Button>
  240. </StepFooter>
  241. );
  242. };
  243. render() {
  244. const {isActive, stepNumber, isCompleted, onEdit} = this.props;
  245. return (
  246. <Panel data-test-id="step-add-data-volume">
  247. <StepHeader
  248. canSkip
  249. title={this.title}
  250. isActive={isActive}
  251. stepNumber={stepNumber}
  252. isCompleted={isCompleted}
  253. onEdit={onEdit}
  254. />
  255. {isActive && this.renderBody()}
  256. {isActive && this.renderFooter()}
  257. </Panel>
  258. );
  259. }
  260. }
  261. export default SetBudgetAndReserves;
  262. const BaseRow = styled('div')`
  263. display: grid;
  264. grid-auto-flow: column;
  265. justify-content: space-between;
  266. align-items: center;
  267. `;
  268. const RowWithTag = styled(BaseRow)`
  269. padding: ${space(2)};
  270. background-color: ${p => p.theme.backgroundSecondary};
  271. `;
  272. // body
  273. const DataVolumeItem = styled(PanelItem)`
  274. display: grid;
  275. grid-auto-flow: row;
  276. gap: ${space(3)};
  277. font-weight: normal;
  278. width: 100%;
  279. margin: 0;
  280. border-bottom: 1px solid ${p => p.theme.innerBorder};
  281. `;
  282. const SectionHeader = styled('div')`
  283. display: grid;
  284. grid-template-columns: repeat(2, auto);
  285. justify-content: space-between;
  286. color: ${p => p.theme.textColor};
  287. font-size: ${p => p.theme.fontSizeExtraLarge};
  288. `;
  289. const Title = styled('label')`
  290. display: grid;
  291. grid-auto-flow: column;
  292. gap: ${space(0.5)};
  293. align-items: center;
  294. margin-bottom: 0px;
  295. font-weight: 600;
  296. `;
  297. const LargeTitle = styled(Title)`
  298. font-size: ${p => p.theme.fontSizeLarge};
  299. `;
  300. const OptionalText = styled('span')`
  301. color: ${p => p.theme.subText};
  302. font-weight: 400;
  303. `;
  304. const Description = styled('div')`
  305. display: grid;
  306. grid-template-columns: repeat(2, auto);
  307. justify-content: space-between;
  308. font-size: ${p => p.theme.fontSizeMedium};
  309. color: ${p => p.theme.gray300};
  310. `;
  311. const Events = styled('div')`
  312. font-size: ${p => p.theme.headerFontSize};
  313. margin: 0;
  314. font-weight: 600;
  315. `;
  316. const MinMax = styled(Description)`
  317. font-size: ${p => p.theme.fontSizeSmall};
  318. `;
  319. // footer
  320. const StepFooter = styled(PanelFooter)`
  321. padding: ${space(2)};
  322. display: grid;
  323. align-items: center;
  324. justify-content: end;
  325. `;