mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
60 lines
1.5 KiB
60 lines
1.5 KiB
import { Modal } from 'ant-design-vue' |
|
import { useClipboard, useI18n } from '#imports' |
|
|
|
export const useCopy = (showDialogIfFailed = false) => { |
|
const { t } = useI18n() |
|
|
|
/** fallback for copy if clipboard api is not supported */ |
|
const copyFallback = async (text: string, retryCount = 0): Promise<boolean> => { |
|
try { |
|
const textAreaEl = document.createElement('textarea') |
|
textAreaEl.innerHTML = text |
|
document.body.appendChild(textAreaEl) |
|
textAreaEl.select() |
|
const result = document.execCommand('copy') |
|
document.body.removeChild(textAreaEl) |
|
if (!result && retryCount < 3) { |
|
// retry if copy failed |
|
return new Promise((resolve, reject) => |
|
setTimeout( |
|
() => |
|
copyFallback(text, retryCount + 1) |
|
.then(resolve) |
|
.catch(reject), |
|
100, |
|
), |
|
) |
|
} |
|
|
|
if (!result) { |
|
throw new Error('failed') |
|
} |
|
return result |
|
} catch (e) { |
|
if (!showDialogIfFailed) throw new Error(t('msg.error.copyToClipboardError')) |
|
|
|
Modal.info({ |
|
title: 'Copy failed, please manually copy it from here', |
|
content: text, |
|
class: 'nc-copy-failed-modal', |
|
width: '550px', |
|
}) |
|
return false |
|
} |
|
} |
|
|
|
const { copy: _copy, isSupported } = useClipboard() |
|
|
|
const copy = async (text: string) => { |
|
try { |
|
if (isSupported.value) { |
|
await _copy(text) |
|
return true |
|
} |
|
} catch {} |
|
|
|
return copyFallback(text) |
|
} |
|
|
|
return { copy } |
|
}
|
|
|