import {Fragment, useEffect, useMemo, useState} from 'react'; import LazyLoad, {forceCheck} from 'react-lazyload'; import {RouteComponentProps} from 'react-router'; import styled from '@emotion/styled'; import {withProfiler} from '@sentry/react'; import debounce from 'lodash/debounce'; import uniqBy from 'lodash/uniqBy'; import {Client} from 'sentry/api'; import {Button} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingError from 'sentry/components/loadingError'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import NoProjectMessage from 'sentry/components/noProjectMessage'; import {PageHeadingQuestionTooltip} from 'sentry/components/pageHeadingQuestionTooltip'; import {useProjectCreationAccess} from 'sentry/components/projects/useProjectCreationAccess'; import SearchBar from 'sentry/components/searchBar'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {DEFAULT_DEBOUNCE_DURATION} from 'sentry/constants'; import {IconAdd, IconUser} from 'sentry/icons'; import {t} from 'sentry/locale'; import ProjectsStatsStore from 'sentry/stores/projectsStatsStore'; import {space} from 'sentry/styles/space'; import {Organization, Project, TeamWithProjects} from 'sentry/types'; import {sortProjects} from 'sentry/utils'; import { onRenderCallback, Profiler, setGroupedEntityTag, } from 'sentry/utils/performanceForSentry'; import useOrganization from 'sentry/utils/useOrganization'; import withApi from 'sentry/utils/withApi'; import withOrganization from 'sentry/utils/withOrganization'; import withTeamsForUser from 'sentry/utils/withTeamsForUser'; import TeamFilter from 'sentry/views/alerts/list/rules/teamFilter'; import ProjectCard from './projectCard'; import Resources from './resources'; import {getTeamParams} from './utils'; type Props = { api: Client; error: Error | null; loadingTeams: boolean; organization: Organization; teams: TeamWithProjects[]; } & RouteComponentProps<{}, {}>; function ProjectCardList({projects}: {projects: Project[]}) { const organization = useOrganization(); const hasProjectAccess = organization.access.includes('project:read'); // By default react-lazyload will only check for intesecting components on scroll // This forceCheck call is necessary to recalculate when filtering projects useEffect(() => { forceCheck(); }, [projects]); return ( {sortProjects(projects).map(project => ( ))} ); } function Dashboard({teams, organization, loadingTeams, error, router, location}: Props) { useEffect(() => { return function cleanup() { ProjectsStatsStore.reset(); }; }, []); const [projectQuery, setProjectQuery] = useState(''); const debouncedSearchQuery = useMemo( () => debounce(handleSearch, DEFAULT_DEBOUNCE_DURATION), [] ); const {canCreateProject} = useProjectCreationAccess({organization}); if (loadingTeams) { return ; } if (error) { return ; } const canJoinTeam = organization.access.includes('team:read'); const selectedTeams = getTeamParams(location ? location.query.team : ''); const filteredTeams = teams.filter(team => selectedTeams.includes(team.id)); const filteredTeamProjects = uniqBy( (filteredTeams ?? teams).flatMap(team => team.projects), 'id' ); const projects = uniqBy( teams.flatMap(teamObj => teamObj.projects), 'id' ); setGroupedEntityTag('projects.total', 1000, projects.length); const currentProjects = selectedTeams.length === 0 ? projects : filteredTeamProjects; const filteredProjects = (currentProjects ?? projects).filter(project => project.slug.includes(projectQuery) ); const showResources = projects.length === 1 && !projects[0].firstEvent; function handleSearch(searchQuery: string) { setProjectQuery(searchQuery); } function handleChangeFilter(activeFilters: string[]) { const {...currentQuery} = location.query; router.push({ pathname: location.pathname, query: { ...currentQuery, team: activeFilters.length > 0 ? activeFilters : '', }, }); } return ( {t('Projects')} {showResources && } ); } function OrganizationDashboard(props: Props) { return ( ); } const SearchAndSelectorWrapper = styled('div')` display: flex; gap: ${space(2)}; justify-content: flex-end; align-items: flex-end; margin-bottom: ${space(2)}; @media (max-width: ${p => p.theme.breakpoints.small}) { display: block; } @media (min-width: ${p => p.theme.breakpoints.xlarge}) { display: flex; } `; const StyledSearchBar = styled(SearchBar)` flex-grow: 1; @media (max-width: ${p => p.theme.breakpoints.small}) { margin-top: ${space(1)}; } `; const ProjectCards = styled('div')` display: grid; gap: ${space(3)}; grid-template-columns: repeat(auto-fill, minmax(1fr, 400px)); @media (min-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: repeat(auto-fill, minmax(470px, 1fr)); } @media (min-width: ${p => p.theme.breakpoints.medium}) { grid-template-columns: repeat(auto-fill, minmax(450px, 1fr)); } `; export {Dashboard}; export default withApi( withOrganization(withTeamsForUser(withProfiler(OrganizationDashboard))) );