import ExternalLink from 'sentry/components/links/externalLink'; import {DEFAULT_QUERY, NEW_DEFAULT_QUERY} from 'sentry/constants'; import {t, tct} from 'sentry/locale'; import type {Organization} from 'sentry/types/organization'; export enum Query { FOR_REVIEW = 'is:unresolved is:for_review assigned_or_suggested:[me, my_teams, none]', // biome-ignore lint/style/useLiteralEnumMembers: Disable for maintenance cost. PRIORITIZED = NEW_DEFAULT_QUERY, UNRESOLVED = 'is:unresolved', IGNORED = 'is:ignored', NEW = 'is:new', ARCHIVED = 'is:archived', ESCALATING = 'is:escalating', REGRESSED = 'is:regressed', REPROCESSING = 'is:reprocessing', } export const CUSTOM_TAB_VALUE = '__custom__'; type OverviewTab = { /** * Emitted analytics event tab name */ analyticsName: string; /** * Will fetch a count to display on this tab */ count: boolean; /** * Tabs can be disabled via flag */ enabled: boolean; name: string; hidden?: boolean; /** * Tooltip text to be hoverable when text has links */ tooltipHoverable?: boolean; /** * Tooltip text for each tab */ tooltipTitle?: React.ReactNode; }; /** * Get a list of currently active tabs */ export function getTabs(organization: Organization) { const hasIssuePriority = organization.features.includes('issue-priority-ui'); const tabs: Array<[string, OverviewTab]> = [ [ Query.PRIORITIZED, { name: t('Prioritized'), analyticsName: 'prioritized', count: true, enabled: hasIssuePriority, }, ], [ Query.UNRESOLVED, { name: t('Unresolved'), analyticsName: 'unresolved', count: true, enabled: !hasIssuePriority, }, ], [ Query.FOR_REVIEW, { name: t('For Review'), analyticsName: 'needs_review', count: true, enabled: true, tooltipTitle: t( 'Issues are marked for review if they are new or escalating, and have not been resolved or archived. Issues are automatically marked reviewed in 7 days.' ), }, ], [ Query.REGRESSED, { name: t('Regressed'), analyticsName: 'regressed', count: true, enabled: true, }, ], [ Query.ESCALATING, { name: t('Escalating'), analyticsName: 'escalating', count: true, enabled: true, }, ], [ Query.ARCHIVED, { name: t('Archived'), analyticsName: 'archived', count: true, enabled: true, }, ], [ Query.IGNORED, { name: t('Ignored'), analyticsName: 'ignored', count: true, enabled: false, tooltipTitle: t(`Ignored issues don’t trigger alerts. When their ignore conditions are met they become Unresolved and are flagged for review.`), }, ], [ Query.REPROCESSING, { name: t('Reprocessing'), analyticsName: 'reprocessing', count: true, enabled: true, tooltipTitle: tct( `These [link:reprocessing issues] will take some time to complete. Any new issues that are created during reprocessing will be flagged for review.`, { link: ( ), } ), tooltipHoverable: true, }, ], [ // Hidden tab to account for custom queries that don't match any of the queries // above. It's necessary because if Tabs's value doesn't match that of any tab item // then Tabs will fall back to a default value, causing unexpected behaviors. CUSTOM_TAB_VALUE, { name: t('Custom'), analyticsName: 'custom', hidden: true, count: false, enabled: true, }, ], ]; return tabs.filter(([_query, tab]) => tab.enabled); } /** * @returns queries that should have counts fetched */ export function getTabsWithCounts(organization: Organization) { const tabs = getTabs(organization); return tabs.filter(([_query, tab]) => tab.count).map(([query]) => query); } export function isForReviewQuery(query: string | undefined) { return !!query && /\bis:for_review\b/.test(query); } // the tab counts will look like 99+ export const TAB_MAX_COUNT = 99; type QueryCount = { count: number; hasMore: boolean; }; export type QueryCounts = Partial>; export enum IssueSortOptions { DATE = 'date', NEW = 'new', TRENDS = 'trends', FREQ = 'freq', USER = 'user', INBOX = 'inbox', } export const DEFAULT_ISSUE_STREAM_SORT = IssueSortOptions.DATE; export function isDefaultIssueStreamSearch( {query, sort}: {query: string; sort: string}, {organization}: {organization: Organization} ) { const defaultQuery = organization.features.includes('issue-priority-ui') ? NEW_DEFAULT_QUERY : DEFAULT_QUERY; return query === defaultQuery && sort === DEFAULT_ISSUE_STREAM_SORT; } export function getSortLabel(key: string) { switch (key) { case IssueSortOptions.NEW: return t('First Seen'); case IssueSortOptions.TRENDS: return t('Trends'); case IssueSortOptions.FREQ: return t('Events'); case IssueSortOptions.USER: return t('Users'); case IssueSortOptions.INBOX: return t('Date Added'); case IssueSortOptions.DATE: default: return t('Last Seen'); } } export const DISCOVER_EXCLUSION_FIELDS: string[] = [ 'query', 'status', 'bookmarked_by', 'assigned', 'assigned_to', 'unassigned', 'subscribed_by', 'active_at', 'first_release', 'first_seen', 'is', '__text', ]; export const FOR_REVIEW_QUERIES: string[] = [Query.FOR_REVIEW]; export const SAVED_SEARCHES_SIDEBAR_OPEN_LOCALSTORAGE_KEY = 'issue-stream-saved-searches-sidebar-open';