rulesPanel.tsx 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. import styled from '@emotion/styled';
  2. import TextArea from 'sentry/components/forms/controls/textarea';
  3. import {Panel, PanelBody, PanelHeader} from 'sentry/components/panels';
  4. import TimeSince from 'sentry/components/timeSince';
  5. import {IconGithub, IconGitlab, IconSentry} from 'sentry/icons';
  6. import {t} from 'sentry/locale';
  7. import {space} from 'sentry/styles/space';
  8. type Props = {
  9. dateUpdated: string | null;
  10. raw: string;
  11. type: 'codeowners' | 'issueowners';
  12. controls?: React.ReactNode[];
  13. 'data-test-id'?: string;
  14. placeholder?: string;
  15. provider?: string;
  16. repoName?: string;
  17. };
  18. function RulesPanel({
  19. raw,
  20. dateUpdated,
  21. provider,
  22. repoName,
  23. type,
  24. placeholder,
  25. controls,
  26. ['data-test-id']: dataTestId,
  27. }: Props) {
  28. function renderIcon() {
  29. switch (provider ?? '') {
  30. case 'github':
  31. return <IconGithub size="sm" />;
  32. case 'gitlab':
  33. return <IconGitlab size="sm" />;
  34. default:
  35. return <IconSentry size="sm" />;
  36. }
  37. }
  38. function renderTitle() {
  39. switch (type) {
  40. case 'codeowners':
  41. return 'CODEOWNERS';
  42. case 'issueowners':
  43. return 'Ownership Rules';
  44. default:
  45. return null;
  46. }
  47. }
  48. return (
  49. <Panel data-test-id={dataTestId}>
  50. <PanelHeader hasButtons>
  51. <Container>
  52. {renderIcon()}
  53. {renderTitle()}
  54. {repoName && <div>{`- ${repoName}`}</div>}
  55. </Container>
  56. <Container>
  57. {dateUpdated && (
  58. <SyncDate>
  59. {t('Last %s', type === 'codeowners' ? t('synced') : t('edited'))}{' '}
  60. <TimeSince date={dateUpdated} />
  61. </SyncDate>
  62. )}
  63. {controls}
  64. </Container>
  65. </PanelHeader>
  66. <PanelBody>
  67. <InnerPanelBody>
  68. <StyledTextArea
  69. monospace
  70. readOnly
  71. value={raw}
  72. spellCheck="false"
  73. autoComplete="off"
  74. autoCorrect="off"
  75. autoCapitalize="off"
  76. placeholder={placeholder}
  77. />
  78. </InnerPanelBody>
  79. </PanelBody>
  80. </Panel>
  81. );
  82. }
  83. export default RulesPanel;
  84. const Container = styled('div')`
  85. display: flex;
  86. align-items: center;
  87. gap: ${space(0.75)};
  88. `;
  89. const InnerPanelBody = styled(PanelBody)`
  90. height: auto;
  91. `;
  92. const StyledTextArea = styled(TextArea)`
  93. height: 350px !important;
  94. overflow: auto;
  95. outline: 0;
  96. width: 100%;
  97. resize: none;
  98. margin: 0;
  99. word-break: break-all;
  100. white-space: pre-wrap;
  101. line-height: ${space(3)};
  102. border: none;
  103. box-shadow: none;
  104. color: transparent;
  105. text-shadow: 0 0 0 #9386a0;
  106. &:hover,
  107. &:focus,
  108. &:active {
  109. border: none;
  110. box-shadow: none;
  111. }
  112. `;
  113. const SyncDate = styled('div')`
  114. font-weight: normal;
  115. text-transform: none;
  116. `;