profileDragDropImport.tsx 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. import * as React from 'react';
  2. import styled from '@emotion/styled';
  3. import {t} from 'sentry/locale';
  4. import {
  5. importDroppedProfile,
  6. ProfileGroup,
  7. } from 'sentry/utils/profiling/profile/importProfile';
  8. export interface ProfileDragDropImportProps {
  9. children: React.ReactNode;
  10. onImport: (profile: ProfileGroup) => void;
  11. }
  12. function ProfileDragDropImport({
  13. onImport,
  14. children,
  15. }: ProfileDragDropImportProps): React.ReactElement {
  16. const [dropState, setDropState] = React.useState<'idle' | 'dragover' | 'processing'>(
  17. 'idle'
  18. );
  19. const [errorMessage, setErrorMessage] = React.useState<string | null>(null);
  20. const onDrop = React.useCallback(
  21. (evt: React.DragEvent<HTMLDivElement>) => {
  22. evt.preventDefault();
  23. evt.stopPropagation();
  24. const file = evt.dataTransfer.items[0].getAsFile();
  25. if (file) {
  26. setDropState('processing');
  27. importDroppedProfile(file)
  28. .then(profile => {
  29. setDropState('idle');
  30. setErrorMessage(null);
  31. onImport(profile);
  32. })
  33. .catch(e => {
  34. setDropState('idle');
  35. setErrorMessage(e.message);
  36. });
  37. }
  38. },
  39. [onImport]
  40. );
  41. const onDragEnter = React.useCallback((evt: React.DragEvent<HTMLDivElement>) => {
  42. evt.preventDefault();
  43. evt.stopPropagation();
  44. setDropState('dragover');
  45. }, []);
  46. const onDragLeave = React.useCallback((evt: React.DragEvent<HTMLDivElement>) => {
  47. evt.preventDefault();
  48. evt.stopPropagation();
  49. setDropState('idle');
  50. }, []);
  51. // This is required to indicate that onDrop is supported on this element
  52. const onDragOver = React.useCallback((evt: React.DragEvent<HTMLDivElement>) => {
  53. evt.preventDefault();
  54. }, []);
  55. return (
  56. <div onDragEnter={onDragEnter}>
  57. {dropState === 'idle' ? null : (
  58. <Overlay onDrop={onDrop} onDragOver={onDragOver} onDragLeave={onDragLeave}>
  59. {t('Drop here')}
  60. <p>{errorMessage}</p>
  61. </Overlay>
  62. )}
  63. {children}
  64. </div>
  65. );
  66. }
  67. const Overlay = styled('div')`
  68. position: absolute;
  69. left: 0;
  70. top: 0;
  71. width: 100%;
  72. height: 100%;
  73. display: grid;
  74. grid: auto/50%;
  75. place-content: center;
  76. z-index: ${p => p.theme.zIndex.modal};
  77. text-align: center;
  78. `;
  79. export {ProfileDragDropImport};