import styled from '@emotion/styled'; import type {Location} from 'history'; import pick from 'lodash/pick'; import moment from 'moment'; import Count from 'sentry/components/count'; import DeprecatedAsyncComponent from 'sentry/components/deprecatedAsyncComponent'; import {normalizeDateTimeParams} from 'sentry/components/organizations/pageFilters/parse'; import * as SidebarSection from 'sentry/components/sidebarSection'; import {URL_PARAM} from 'sentry/constants/pageFilters'; import {t, tn} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {CrashFreeTimeBreakdown, Organization} from 'sentry/types'; import {defined} from 'sentry/utils'; import {displayCrashFreePercent} from '../../../utils'; type Props = DeprecatedAsyncComponent['props'] & { location: Location; organization: Organization; projectSlug: string; version: string; }; type State = DeprecatedAsyncComponent['state'] & { releaseStats?: {usersBreakdown: CrashFreeTimeBreakdown} | null; }; class TotalCrashFreeUsers extends DeprecatedAsyncComponent { shouldReload = true; getEndpoints(): ReturnType { const {location, organization, projectSlug, version} = this.props; return [ [ 'releaseStats', `/projects/${organization.slug}/${projectSlug}/releases/${encodeURIComponent( version )}/stats/`, { query: { ...normalizeDateTimeParams( pick(location.query, [URL_PARAM.PROJECT, URL_PARAM.ENVIRONMENT]) ), type: 'sessions', }, }, ], ]; } componentDidUpdate(prevProps: Props) { if (prevProps.version !== this.props.version) { this.remountComponent(); } } renderLoading() { return this.renderBody(); } renderBody() { const crashFreeTimeBreakdown = this.state.releaseStats?.usersBreakdown; if (!crashFreeTimeBreakdown?.length) { return null; } const timeline = crashFreeTimeBreakdown .map(({date, crashFreeUsers, totalUsers}, index, data) => { // count number of crash free users from knowing percent and total const crashFreeUserCount = Math.round(((crashFreeUsers ?? 0) * totalUsers) / 100); // first item of timeline is release creation date, then we want to have relative date label const dateLabel = index === 0 ? t('Release created') : `${moment(data[0].date).from(date, true)} ${t('later')}`; return {date: moment(date), dateLabel, crashFreeUsers, crashFreeUserCount}; }) // remove those timeframes that are in the future .filter(item => item.date.isBefore()) // we want timeline to go from bottom to up .reverse(); if (!timeline.length) { return null; } return ( {t('Total Crash Free Users')} {timeline.map(row => ( {row.date.format('MMMM D')} {' '} {tn('user', 'users', row.crashFreeUserCount)} {row.dateLabel} {defined(row.crashFreeUsers) ? displayCrashFreePercent(row.crashFreeUsers) : '-'} ))} ); } } const Timeline = styled('div')` font-size: ${p => p.theme.fontSizeMedium}; line-height: 1.2; `; const DOT_SIZE = 10; const Row = styled('div')` border-left: 1px solid ${p => p.theme.border}; padding-left: ${space(2)}; padding-bottom: ${space(1)}; margin-left: ${space(1)}; position: relative; &:before { content: ''; width: ${DOT_SIZE}px; height: ${DOT_SIZE}px; border-radius: 100%; background-color: ${p => p.theme.purple300}; position: absolute; top: 0; left: -${Math.floor(DOT_SIZE / 2)}px; } &:last-child { border-left: 0; } `; const InnerRow = styled('div')` display: grid; grid-column-gap: ${space(2)}; grid-auto-flow: column; grid-auto-columns: 1fr; padding-bottom: ${space(0.5)}; `; const Text = styled('div')<{bold?: boolean; right?: boolean}>` text-align: ${p => (p.right ? 'right' : 'left')}; color: ${p => (p.bold ? p.theme.textColor : p.theme.gray300)}; padding-bottom: ${space(0.25)}; ${p => p.theme.overflowEllipsis}; `; const Percent = styled(Text)` font-variant-numeric: tabular-nums; `; export default TotalCrashFreeUsers;