import type {InjectedRouter} from 'react-router'; import styled from '@emotion/styled'; import type {Location} from 'history'; import pick from 'lodash/pick'; import {createDashboard} from 'sentry/actionCreators/dashboards'; import {addSuccessMessage} from 'sentry/actionCreators/indicator'; import {openImportDashboardFromFileModal} from 'sentry/actionCreators/modal'; import type {Client} from 'sentry/api'; import Feature from 'sentry/components/acl/feature'; import {Alert} from 'sentry/components/alert'; import {Button} from 'sentry/components/button'; import ButtonBar from 'sentry/components/buttonBar'; import {CompactSelect} from 'sentry/components/compactSelect'; import * as Layout from 'sentry/components/layouts/thirds'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import NoProjectMessage from 'sentry/components/noProjectMessage'; import {PageHeadingQuestionTooltip} from 'sentry/components/pageHeadingQuestionTooltip'; import SearchBar from 'sentry/components/searchBar'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import Switch from 'sentry/components/switchButton'; import {IconAdd} from 'sentry/icons'; import {t} from 'sentry/locale'; import {space} from 'sentry/styles/space'; import type {SelectValue} from 'sentry/types/core'; import type {Organization} from 'sentry/types/organization'; import {trackAnalytics} from 'sentry/utils/analytics'; import {browserHistory} from 'sentry/utils/browserHistory'; import {decodeScalar} from 'sentry/utils/queryString'; import normalizeUrl from 'sentry/utils/url/normalizeUrl'; import withApi from 'sentry/utils/withApi'; import withOrganization from 'sentry/utils/withOrganization'; import {DashboardImportButton} from 'sentry/views/dashboards/manage/dashboardImport'; import DeprecatedAsyncView from 'sentry/views/deprecatedAsyncView'; import {DASHBOARDS_TEMPLATES} from '../data'; import {assignDefaultLayout, getInitialColumnDepths} from '../layoutUtils'; import type {DashboardDetails, DashboardListItem} from '../types'; import DashboardList from './dashboardList'; import TemplateCard from './templateCard'; import {setShowTemplates, shouldShowTemplates} from './utils'; const SORT_OPTIONS: SelectValue[] = [ {label: t('My Dashboards'), value: 'mydashboards'}, {label: t('Dashboard Name (A-Z)'), value: 'title'}, {label: t('Date Created (Newest)'), value: '-dateCreated'}, {label: t('Date Created (Oldest)'), value: 'dateCreated'}, {label: t('Most Popular'), value: 'mostPopular'}, {label: t('Recently Viewed'), value: 'recentlyViewed'}, ]; type Props = { api: Client; location: Location; organization: Organization; router: InjectedRouter; } & DeprecatedAsyncView['props']; type State = { dashboards: DashboardListItem[] | null; dashboardsPageLinks: string; showTemplates: boolean; } & DeprecatedAsyncView['state']; class ManageDashboards extends DeprecatedAsyncView { getDefaultState() { return { ...super.getDefaultState(), showTemplates: shouldShowTemplates(), }; } getEndpoints(): ReturnType { const {organization, location} = this.props; const endpoints: ReturnType = [ [ 'dashboards', `/organizations/${organization.slug}/dashboards/`, { query: { ...pick(location.query, ['cursor', 'query']), sort: this.getActiveSort().value, per_page: '9', }, }, ], ]; return endpoints; } getActiveSort() { const {location} = this.props; const urlSort = decodeScalar(location.query.sort, 'mydashboards'); return SORT_OPTIONS.find(item => item.value === urlSort) || SORT_OPTIONS[0]; } onDashboardsChange() { this.reloadData(); } handleSearch(query: string) { const {location, router, organization} = this.props; trackAnalytics('dashboards_manage.search', { organization, }); router.push({ pathname: location.pathname, query: {...location.query, cursor: undefined, query}, }); } handleSortChange = (value: string) => { const {location, organization} = this.props; trackAnalytics('dashboards_manage.change_sort', { organization, sort: value, }); browserHistory.push({ pathname: location.pathname, query: { ...location.query, cursor: undefined, sort: value, }, }); }; toggleTemplates = () => { const {showTemplates} = this.state; const {organization} = this.props; trackAnalytics('dashboards_manage.templates.toggle', { organization, show_templates: !showTemplates, }); this.setState({showTemplates: !showTemplates}, () => { setShowTemplates(!showTemplates); }); }; getQuery() { const {query} = this.props.location.query; return typeof query === 'string' ? query : undefined; } renderTemplates() { return ( {DASHBOARDS_TEMPLATES.map(dashboard => ( this.onPreview(dashboard.id)} onAdd={() => this.onAdd(dashboard)} key={dashboard.title} /> ))} ); } renderActions() { const activeSort = this.getActiveSort(); return ( this.handleSearch(query)} /> this.handleSortChange(opt.value)} position="bottom-end" /> ); } renderNoAccess() { return ( {t("You don't have access to this feature")} ); } renderDashboards() { const {dashboards, dashboardsPageLinks} = this.state; const {organization, location, api} = this.props; return ( this.onDashboardsChange()} /> ); } getTitle() { return t('Dashboards'); } onCreate() { const {organization, location} = this.props; trackAnalytics('dashboards_manage.create.start', { organization, }); browserHistory.push( normalizeUrl({ pathname: `/organizations/${organization.slug}/dashboards/new/`, query: location.query, }) ); } async onAdd(dashboard: DashboardDetails) { const {organization, api} = this.props; trackAnalytics('dashboards_manage.templates.add', { organization, dashboard_id: dashboard.id, dashboard_title: dashboard.title, was_previewed: false, }); const newDashboard = await createDashboard( api, organization.slug, { ...dashboard, widgets: assignDefaultLayout(dashboard.widgets, getInitialColumnDepths()), }, true ); addSuccessMessage(`${dashboard.title} dashboard template successfully added.`); this.loadDashboard(newDashboard.id); } loadDashboard(dashboardId: string) { const {organization, location} = this.props; browserHistory.push( normalizeUrl({ pathname: `/organizations/${organization.slug}/dashboards/${dashboardId}/`, query: location.query, }) ); } onPreview(dashboardId: string) { const {organization, location} = this.props; trackAnalytics('dashboards_manage.templates.preview', { organization, dashboard_id: dashboardId, }); browserHistory.push( normalizeUrl({ pathname: `/organizations/${organization.slug}/dashboards/new/${dashboardId}/`, query: location.query, }) ); } renderLoading() { return ( ); } renderBody() { const {showTemplates} = this.state; const {organization, api, location} = this.props; return ( {t('Dashboards')} {t('Show Templates')} {showTemplates && this.renderTemplates()} {this.renderActions()} {this.renderDashboards()} ); } } const StyledActions = styled('div')` display: grid; grid-template-columns: auto max-content; gap: ${space(2)}; margin-bottom: ${space(2)}; @media (max-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: auto; } `; const TemplateSwitch = styled('label')` font-weight: ${p => p.theme.fontWeightNormal}; font-size: ${p => p.theme.fontSizeLarge}; display: flex; align-items: center; gap: ${space(1)}; width: max-content; margin: 0; `; const TemplateContainer = styled('div')` display: grid; gap: ${space(2)}; margin-bottom: ${space(0.5)}; @media (min-width: ${p => p.theme.breakpoints.small}) { grid-template-columns: repeat(2, minmax(200px, 1fr)); } @media (min-width: ${p => p.theme.breakpoints.large}) { grid-template-columns: repeat(4, minmax(200px, 1fr)); } `; export default withApi(withOrganization(ManageDashboards));