import {Fragment, useCallback, useEffect, useRef, useState} from 'react'; import styled from '@emotion/styled'; import {bulkUpdate} from 'sentry/actionCreators/group'; import {addErrorMessage} from 'sentry/actionCreators/indicator'; import {closeModal, ModalRenderProps} from 'sentry/actionCreators/modal'; import AutoSelectText from 'sentry/components/autoSelectText'; import Button from 'sentry/components/button'; import LoadingIndicator from 'sentry/components/loadingIndicator'; import Switch from 'sentry/components/switchButton'; import {IconCopy, IconRefresh} from 'sentry/icons'; import {t} from 'sentry/locale'; import GroupStore from 'sentry/stores/groupStore'; import {useLegacyStore} from 'sentry/stores/useLegacyStore'; import space from 'sentry/styles/space'; import type {Group, Organization} from 'sentry/types'; import useApi from 'sentry/utils/useApi'; interface ShareIssueModalProps extends ModalRenderProps { groupId: string; onToggle: () => void; organization: Organization; projectSlug: string; disabled?: boolean; disabledReason?: string; } type UrlRef = React.ElementRef; function ShareIssueModal({ Header, Body, Footer, organization, projectSlug, groupId, onToggle, }: ShareIssueModalProps) { const api = useApi({persistInFlight: true}); const [loading, setLoading] = useState(false); const urlRef = useRef(null); const groups = useLegacyStore(GroupStore); const group = (groups as Group[]).find(item => item.id === groupId)!; const isShared = group.isPublic; function getShareUrl() { const path = `/share/issue/${group.shareId}/`; const {host, protocol} = window.location; return `${protocol}//${host}${path}`; } const shareUrl = group.shareId ? getShareUrl() : null; const handleShare = useCallback( (e: React.MouseEvent | null, reshare?: boolean) => { e?.preventDefault(); setLoading(true); onToggle(); bulkUpdate( api, { orgId: organization.slug, projectId: projectSlug, itemIds: [groupId], data: { isPublic: reshare ?? !isShared, }, }, { error: () => { addErrorMessage(t('Error sharing')); }, complete: () => { setLoading(false); }, } ); }, [api, setLoading, onToggle, isShared, organization.slug, projectSlug, groupId] ); /** * Share as soon as modal is opened */ useEffect(() => { if (isShared) { return; } handleShare(null, true); // eslint-disable-next-line react-hooks/exhaustive-deps -- we only want to run this on open }, []); return (

{t('Share Issue')}

{t('Create a public link')} {t('Share a link with anyone outside your organization')}
{loading && ( )} {!loading && isShared && shareUrl && ( {shareUrl} { navigator.clipboard.writeText(shareUrl); urlRef.current?.selectText(); }} icon={} aria-label={t('Copy to clipboard')} /> } onClick={() => handleShare(null, true)} /> )}
); } export default ShareIssueModal; /** * min-height reduces layout shift when switching on and off */ const ModalContent = styled('div')` display: flex; gap: ${space(2)}; flex-direction: column; min-height: 100px; `; const SwitchWrapper = styled('div')` display: flex; justify-content: space-between; align-items: center; gap: ${space(2)}; `; const Title = styled('div')` padding-right: ${space(4)}; white-space: nowrap; `; const SubText = styled('p')` color: ${p => p.theme.subText}; font-size: ${p => p.theme.fontSizeSmall}; `; const LoadingContainer = styled('div')` display: flex; justify-content: center; `; const UrlContainer = styled('div')` display: flex; align-items: stretch; border: 1px solid ${p => p.theme.border}; border-radius: ${space(0.5)}; `; const StyledAutoSelectText = styled(AutoSelectText)` padding: ${space(1)} ${space(1)}; ${p => p.theme.overflowEllipsis} `; const TextContainer = styled('div')` position: relative; display: flex; flex-grow: 1; background-color: transparent; border-right: 1px solid ${p => p.theme.border}; min-width: 0; `; const ClipboardButton = styled(Button)` border-radius: 0; border-right: 1px solid ${p => p.theme.border}; height: 100%; flex-shrink: 0; &:hover { border-right: 1px solid ${p => p.theme.border}; } `; const ReshareButton = styled(Button)` height: 100%; flex-shrink: 0; `;