123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- // eslint-disable-next-line no-restricted-imports
- import {withRouter, WithRouterProps} from 'react-router';
- import styled from '@emotion/styled';
- import {updateEnvironments} from 'sentry/actionCreators/pageFilters';
- import Badge from 'sentry/components/badge';
- import type {ButtonProps} from 'sentry/components/button';
- import EnvironmentSelector from 'sentry/components/organizations/environmentSelector';
- import PageFilterDropdownButton from 'sentry/components/organizations/pageFilters/pageFilterDropdownButton';
- import PageFilterPinIndicator from 'sentry/components/organizations/pageFilters/pageFilterPinIndicator';
- import {IconWindow} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import space from 'sentry/styles/space';
- import {trimSlug} from 'sentry/utils/trimSlug';
- import useOrganization from 'sentry/utils/useOrganization';
- import usePageFilters from 'sentry/utils/usePageFilters';
- import useProjects from 'sentry/utils/useProjects';
- type EnvironmentSelectorProps = React.ComponentProps<typeof EnvironmentSelector>;
- type Props = {
- router: WithRouterProps['router'];
- alignDropdown?: EnvironmentSelectorProps['alignDropdown'];
- disabled?: EnvironmentSelectorProps['disabled'];
- /**
- * Max character length for the dropdown title. Default is 20. This number
- * is used to determine how many projects to show, and how much to truncate.
- */
- maxTitleLength?: number;
- /**
- * Reset these URL params when we fire actions (custom routing only)
- */
- resetParamsOnChange?: string[];
- size?: ButtonProps['size'];
- };
- function EnvironmentPageFilter({
- router,
- resetParamsOnChange = [],
- alignDropdown,
- disabled,
- maxTitleLength = 20,
- size = 'md',
- }: Props) {
- const {projects, initiallyLoaded: projectsLoaded} = useProjects();
- const organization = useOrganization();
- const {selection, isReady, desyncedFilters} = usePageFilters();
- const handleUpdateEnvironments = (environments: string[]) => {
- updateEnvironments(environments, router, {
- save: true,
- resetParams: resetParamsOnChange,
- });
- };
- const customDropdownButton: EnvironmentSelectorProps['customDropdownButton'] = ({
- isOpen,
- value,
- }) => {
- const environmentsToShow =
- value[0]?.length + value[1]?.length <= maxTitleLength - 2
- ? value.slice(0, 2)
- : value.slice(0, 1);
- const summary = value.length
- ? environmentsToShow.map(env => trimSlug(env, maxTitleLength)).join(', ')
- : t('All Envs');
- return (
- <PageFilterDropdownButton
- isOpen={isOpen}
- highlighted={desyncedFilters.has('environments')}
- data-test-id="page-filter-environment-selector"
- disabled={disabled}
- size={size}
- >
- <DropdownTitle>
- <PageFilterPinIndicator filter="environments">
- <IconWindow />
- </PageFilterPinIndicator>
- <TitleContainer>
- {summary}
- {!!value.length && value.length > environmentsToShow.length && (
- <Badge text={`+${value.length - environmentsToShow.length}`} />
- )}
- </TitleContainer>
- </DropdownTitle>
- </PageFilterDropdownButton>
- );
- };
- const customLoadingIndicator = (
- <PageFilterDropdownButton
- showChevron={false}
- disabled
- data-test-id="page-filter-environment-selector"
- >
- <DropdownTitle>
- <IconWindow />
- <TitleContainer>{t('Loading\u2026')}</TitleContainer>
- </DropdownTitle>
- </PageFilterDropdownButton>
- );
- return (
- <EnvironmentSelector
- organization={organization}
- projects={projects}
- loadingProjects={!projectsLoaded || !isReady}
- selectedProjects={selection.projects}
- value={selection.environments}
- onUpdate={handleUpdateEnvironments}
- customDropdownButton={customDropdownButton}
- customLoadingIndicator={customLoadingIndicator}
- alignDropdown={alignDropdown}
- disabled={disabled}
- />
- );
- }
- const TitleContainer = styled('div')`
- display: flex;
- align-items: center;
- justify-content: flex-start;
- flex: 1 1 0%;
- margin-left: ${space(1)};
- overflow: hidden;
- white-space: nowrap;
- text-overflow: ellipsis;
- `;
- const DropdownTitle = styled('div')`
- display: flex;
- align-items: center;
- flex: 1;
- `;
- export default withRouter<Props>(EnvironmentPageFilter);
|