UNPKG

@mui/x-data-grid

Version:

The Community plan edition of the Data Grid components (MUI X).

97 lines (94 loc) 3.63 kB
import * as React from 'react'; import { useGridEventPriority, useGridNativeEventListener } from "../../utils/index.js"; import { gridFocusCellSelector } from "../focus/gridFocusStateSelector.js"; import { serializeCellValue } from "../export/serializers/csvSerializer.js"; import { isCopyShortcut } from "../../../utils/keyboardUtils.js"; import { gridRowSelectionCountSelector } from "../rowSelection/index.js"; function writeToClipboardPolyfill(data) { const span = document.createElement('span'); span.style.whiteSpace = 'pre'; span.style.userSelect = 'all'; span.style.opacity = '0px'; span.textContent = data; document.body.appendChild(span); const range = document.createRange(); range.selectNode(span); const selection = window.getSelection(); selection.removeAllRanges(); selection.addRange(range); try { document.execCommand('copy'); } finally { document.body.removeChild(span); } } function copyToClipboard(data) { if (navigator.clipboard) { navigator.clipboard.writeText(data).catch(() => { writeToClipboardPolyfill(data); }); } else { writeToClipboardPolyfill(data); } } function hasNativeSelection(element) { // When getSelection is called on an <iframe> that is not displayed Firefox will return null. if (window.getSelection()?.toString()) { return true; } // window.getSelection() returns an empty string in Firefox for selections inside a form element. // See: https://bugzilla.mozilla.org/show_bug.cgi?id=85686. // Instead, we can use element.selectionStart that is only defined on form elements. if (element && (element.selectionEnd || 0) - (element.selectionStart || 0) > 0) { return true; } return false; } /** * @requires useGridCsvExport (method) * @requires useGridSelection (method) */ export const useGridClipboard = (apiRef, props) => { const ignoreValueFormatterProp = props.ignoreValueFormatterDuringExport; const ignoreValueFormatter = (typeof ignoreValueFormatterProp === 'object' ? ignoreValueFormatterProp?.clipboardExport : ignoreValueFormatterProp) || false; const clipboardCopyCellDelimiter = props.clipboardCopyCellDelimiter; const handleCopy = React.useCallback(event => { if (!isCopyShortcut(event)) { return; } // Do nothing if there's a native selection if (hasNativeSelection(event.target)) { return; } let textToCopy = ''; const selectedRowsCount = gridRowSelectionCountSelector(apiRef); if (selectedRowsCount > 0) { textToCopy = apiRef.current.getDataAsCsv({ includeHeaders: false, delimiter: clipboardCopyCellDelimiter, shouldAppendQuotes: false, escapeFormulas: false }); } else { const focusedCell = gridFocusCellSelector(apiRef); if (focusedCell) { const cellParams = apiRef.current.getCellParams(focusedCell.id, focusedCell.field); textToCopy = serializeCellValue(cellParams, { csvOptions: { delimiter: clipboardCopyCellDelimiter, shouldAppendQuotes: false, escapeFormulas: false }, ignoreValueFormatter }); } } textToCopy = apiRef.current.unstable_applyPipeProcessors('clipboardCopy', textToCopy); if (textToCopy) { copyToClipboard(textToCopy); apiRef.current.publishEvent('clipboardCopy', textToCopy); } }, [apiRef, ignoreValueFormatter, clipboardCopyCellDelimiter]); useGridNativeEventListener(apiRef, () => apiRef.current.rootElementRef.current, 'keydown', handleCopy); useGridEventPriority(apiRef, 'clipboardCopy', props.onClipboardCopy); };