Browse Source

Merge pull request #5065 from nocodb/fix/3598-copy-to-clipboard

fix: if copy failed show error message and show content in dialog(optional)
pull/5071/head
Raju Udava 2 years ago committed by GitHub
parent
commit
7f4991f77c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 7
      packages/nc-gui/assets/style.scss
  2. 14
      packages/nc-gui/components/account/Token.vue
  3. 15
      packages/nc-gui/components/account/UserList.vue
  4. 12
      packages/nc-gui/components/account/UsersModal.vue
  5. 12
      packages/nc-gui/components/smartsheet/ApiSnippet.vue
  6. 23
      packages/nc-gui/components/smartsheet/toolbar/ShareView.vue
  7. 10
      packages/nc-gui/components/smartsheet/toolbar/SharedViewList.vue
  8. 19
      packages/nc-gui/components/tabs/auth/ApiTokenManagement.vue
  9. 13
      packages/nc-gui/components/tabs/auth/UserManagement.vue
  10. 25
      packages/nc-gui/components/tabs/auth/user-management/ShareBase.vue
  11. 12
      packages/nc-gui/components/tabs/auth/user-management/UsersModal.vue
  12. 16
      packages/nc-gui/components/template/Editor.vue
  13. 60
      packages/nc-gui/composables/useCopy.ts
  14. 29
      packages/nc-gui/pages/[projectType]/[projectId]/index.vue
  15. 10
      packages/nc-gui/pages/index/index/index.vue

7
packages/nc-gui/assets/style.scss

@ -311,3 +311,10 @@ a {
@apply block bg-red-500 @apply block bg-red-500
} }
} }
.nc-copy-failed-modal .ant-modal-confirm-content{
@apply pt-4 overflow-auto;
white-space: pre;
user-select: auto;
}

14
packages/nc-gui/components/account/Token.vue

@ -81,14 +81,18 @@ const generateToken = async () => {
$e('a:api-token:generate') $e('a:api-token:generate')
} }
const copyToken = (token: string | undefined) => { const copyToken = async (token: string | undefined) => {
if (!token) return if (!token) return
copy(token) try {
// Copied to clipboard await copy(token)
message.info(t('msg.info.copiedToClipboard')) // Copied to clipboard
message.info(t('msg.info.copiedToClipboard'))
$e('c:api-token:copy') $e('c:api-token:copy')
} catch (e) {
message.error(e.message)
}
} }
const descriptionInput = (el) => { const descriptionInput = (el) => {

15
packages/nc-gui/components/account/UserList.vue

@ -99,13 +99,16 @@ const resendInvite = async (user: User) => {
$e('a:org-user:resend-invite') $e('a:org-user:resend-invite')
} }
const copyInviteUrl = (user: User) => { const copyInviteUrl = async (user: User) => {
if (!user.invite_token) return if (!user.invite_token) return
try {
await copy(`${dashboardUrl}#/signup/${user.invite_token}`)
copy(`${dashboardUrl}#/signup/${user.invite_token}`) // Invite URL copied to clipboard
message.success(t('msg.success.inviteURLCopied'))
// Invite URL copied to clipboard } catch (e) {
message.success(t('msg.success.inviteURLCopied')) message.error(e.message)
}
$e('c:user:copy-url') $e('c:user:copy-url')
} }
@ -113,7 +116,7 @@ const copyPasswordResetUrl = async (user: User) => {
try { try {
const { reset_password_url } = await api.orgUsers.generatePasswordResetToken(user.id) const { reset_password_url } = await api.orgUsers.generatePasswordResetToken(user.id)
copy(reset_password_url) await copy(reset_password_url!)
// Invite URL copied to clipboard // Invite URL copied to clipboard
message.success(t('msg.success.passwordResetURLCopied')) message.success(t('msg.success.passwordResetURLCopied'))

12
packages/nc-gui/components/account/UsersModal.vue

@ -93,12 +93,14 @@ const inviteUrl = $computed(() => (usersData.invitationToken ? `${dashboardUrl}#
const copyUrl = async () => { const copyUrl = async () => {
if (!inviteUrl) return if (!inviteUrl) return
try {
await copy(inviteUrl)
await copy(inviteUrl) // Copied shareable base url to clipboard!
message.success(t('msg.success.shareableURLCopied'))
// Copied shareable base url to clipboard! } catch (e) {
message.success(t('msg.success.shareableURLCopied')) message.error(e.message)
}
$e('c:shared-base:copy-url') $e('c:shared-base:copy-url')
} }

12
packages/nc-gui/components/smartsheet/ApiSnippet.vue

@ -126,10 +126,14 @@ api.dbViewRow.list(
return snippet.convert(activeLang?.name, selectedClient || (activeLang?.clients && activeLang?.clients[0]), {}) return snippet.convert(activeLang?.name, selectedClient || (activeLang?.clients && activeLang?.clients[0]), {})
}) })
const onCopyToClipboard = () => { const onCopyToClipboard = async () => {
copy(code) try {
// Copied to clipboard await copy(code)
message.info(t('msg.info.copiedToClipboard')) // Copied to clipboard
message.info(t('msg.info.copiedToClipboard'))
} catch (e) {
message.error(e.message)
}
} }
const afterVisibleChange = (visible: boolean) => { const afterVisibleChange = (visible: boolean) => {

23
packages/nc-gui/components/smartsheet/toolbar/ShareView.vue

@ -186,10 +186,14 @@ function onChangeTheme(color: string) {
const copyLink = async () => { const copyLink = async () => {
if (sharedViewUrl.value) { if (sharedViewUrl.value) {
await copy(sharedViewUrl.value) try {
await copy(sharedViewUrl.value)
// Copied to clipboard // Copied to clipboard
message.success(t('msg.info.copiedToClipboard')) message.success(t('msg.info.copiedToClipboard'))
} catch (e) {
message.error(e.message)
}
} }
} }
@ -217,10 +221,14 @@ const iframeCode = computed(() => {
const copyIframeCode = async () => { const copyIframeCode = async () => {
if (iframeCode.value) { if (iframeCode.value) {
await copy(iframeCode.value) try {
await copy(iframeCode.value)
// Copied to clipboard // Copied to clipboard
message.success(t('msg.info.copiedToClipboard')) message.success(t('msg.info.copiedToClipboard'))
} catch (e) {
message.error(e.message)
}
} }
} }
</script> </script>
@ -268,7 +276,8 @@ const copyIframeCode = async () => {
class="flex gap-1 items-center pb-1 text-gray-500 cursor-pointer font-weight-medium mb-2 mt-4 pl-1" class="flex gap-1 items-center pb-1 text-gray-500 cursor-pointer font-weight-medium mb-2 mt-4 pl-1"
@click="copyIframeCode" @click="copyIframeCode"
> >
<MdiCodeTags class="text-gray-500" /> Embed this view in your site <MdiCodeTags class="text-gray-500" />
Embed this view in your site
</div> </div>
<div class="px-1 mt-2 flex flex-col gap-3"> <div class="px-1 mt-2 flex flex-col gap-3">

10
packages/nc-gui/components/smartsheet/toolbar/SharedViewList.vue

@ -75,9 +75,13 @@ const renderAllowCSVDownload = (view: SharedViewType) => {
} }
const copyLink = (view: SharedViewType) => { const copyLink = (view: SharedViewType) => {
copy(`${dashboardUrl?.value as string}#${sharedViewUrl(view)}`) try {
// Copied to clipboard copy(`${dashboardUrl?.value as string}#${sharedViewUrl(view)}`)
message.success(t('msg.info.copiedToClipboard')) // Copied to clipboard
message.success(t('msg.info.copiedToClipboard'))
} catch (e) {
message.error(e.message)
}
} }
const deleteLink = async (id: string) => { const deleteLink = async (id: string) => {

19
packages/nc-gui/components/tabs/auth/ApiTokenManagement.vue

@ -33,13 +33,16 @@ const openNewTokenModal = () => {
$e('c:api-token:generate') $e('c:api-token:generate')
} }
const copyToken = (token: string | undefined) => { const copyToken = async (token: string | undefined) => {
if (!token) return if (!token) return
copy(token) try {
// Copied to clipboard await copy(token)
message.info(t('msg.info.copiedToClipboard')) // Copied to clipboard
message.info(t('msg.info.copiedToClipboard'))
} catch (e) {
message.error(e.message)
}
$e('c:api-token:copy') $e('c:api-token:copy')
} }
@ -145,8 +148,8 @@ onMounted(() => {
<div class="flex flex-row justify-center mt-2 text-center w-full text-base">This action will remove this API Token</div> <div class="flex flex-row justify-center mt-2 text-center w-full text-base">This action will remove this API Token</div>
<div class="flex mt-6 justify-center space-x-2"> <div class="flex mt-6 justify-center space-x-2">
<a-button @click="showDeleteTokenModal = false"> {{ $t('general.cancel') }} </a-button> <a-button @click="showDeleteTokenModal = false"> {{ $t('general.cancel') }}</a-button>
<a-button type="primary" danger @click="deleteToken()"> {{ $t('general.confirm') }} </a-button> <a-button type="primary" danger @click="deleteToken()"> {{ $t('general.confirm') }}</a-button>
</div> </div>
</div> </div>
</a-modal> </a-modal>
@ -205,7 +208,7 @@ onMounted(() => {
</a-tooltip> </a-tooltip>
<a-tooltip placement="bottom"> <a-tooltip placement="bottom">
<template #title> {{ $t('general.copy') }} </template> <template #title> {{ $t('general.copy') }}</template>
<a-button type="text" class="!rounded-md" @click="copyToken(item.token)"> <a-button type="text" class="!rounded-md" @click="copyToken(item.token)">
<template #icon> <template #icon>

13
packages/nc-gui/components/tabs/auth/UserManagement.vue

@ -140,13 +140,16 @@ const resendInvite = async (user: User) => {
$e('a:user:resend-invite') $e('a:user:resend-invite')
} }
const copyInviteUrl = (user: User) => { const copyInviteUrl = async (user: User) => {
if (!user.invite_token) return if (!user.invite_token) return
try {
await copy(`${dashboardUrl}#/signup/${user.invite_token}`)
copy(`${dashboardUrl}#/signup/${user.invite_token}`) // Invite URL copied to clipboard
message.success(t('msg.success.inviteURLCopied'))
// Invite URL copied to clipboard } catch (e) {
message.success(t('msg.success.inviteURLCopied')) message.error(e.message)
}
$e('c:user:copy-url') $e('c:user:copy-url')
} }

25
packages/nc-gui/components/tabs/auth/user-management/ShareBase.vue

@ -94,11 +94,14 @@ const recreate = async () => {
const copyUrl = async () => { const copyUrl = async () => {
if (!url) return if (!url) return
try {
await copy(url)
await copy(url) // Copied shareable base url to clipboard!
message.success(t('msg.success.shareableURLCopied'))
// Copied shareable base url to clipboard! } catch (e) {
message.success(t('msg.success.shareableURLCopied')) message.error(e.message)
}
$e('c:shared-base:copy-url') $e('c:shared-base:copy-url')
} }
@ -111,10 +114,10 @@ const navigateToSharedBase = () => {
$e('c:shared-base:open-url') $e('c:shared-base:open-url')
} }
const generateEmbeddableIframe = () => { const generateEmbeddableIframe = async () => {
if (!url) return if (!url) return
try {
copy(`<iframe await copy(`<iframe
class="nc-embed" class="nc-embed"
src="${url}?embed" src="${url}?embed"
frameborder="0" frameborder="0"
@ -122,9 +125,11 @@ width="100%"
height="700" height="700"
style="background: transparent; border: 1px solid #ddd"></iframe>`) style="background: transparent; border: 1px solid #ddd"></iframe>`)
// Copied embeddable html code! // Copied embeddable html code!
message.success(t('msg.success.embeddableHTMLCodeCopied')) message.success(t('msg.success.embeddableHTMLCodeCopied'))
} catch (e) {
message.error(e.message)
}
$e('c:shared-base:copy-embed-frame') $e('c:shared-base:copy-embed-frame')
} }

12
packages/nc-gui/components/tabs/auth/user-management/UsersModal.vue

@ -125,12 +125,14 @@ const inviteUrl = $computed(() => (usersData.invitationToken ? `${dashboardUrl}#
const copyUrl = async () => { const copyUrl = async () => {
if (!inviteUrl) return if (!inviteUrl) return
try {
await copy(inviteUrl)
await copy(inviteUrl) // Copied shareable base url to clipboard!
message.success(t('msg.success.shareableURLCopied'))
// Copied shareable base url to clipboard! } catch (e) {
message.success(t('msg.success.shareableURLCopied')) message.error(e.message)
}
$e('c:shared-base:copy-url') $e('c:shared-base:copy-url')
} }

16
packages/nc-gui/components/template/Editor.vue

@ -501,16 +501,12 @@ async function importTemplate() {
} }
} }
} }
const createdTable = await $api.base.tableCreate( const createdTable = await $api.base.tableCreate(project.value?.id as string, (baseId || project.value?.bases?.[0].id)!, {
project.value?.id as string, table_name: table.table_name,
(baseId || project.value?.bases?.[0].id)!, // leave title empty to get a generated one based on table_name
{ title: '',
table_name: table.table_name, columns: table.columns || [],
// leave title empty to get a generated one based on table_name })
title: '',
columns: table.columns || [],
},
)
table.id = createdTable.id table.id = createdTable.id
table.title = createdTable.title table.title = createdTable.title

60
packages/nc-gui/composables/useCopy.ts

@ -1,15 +1,46 @@
import { useClipboard } from '#imports' import { Modal } from 'ant-design-vue'
import { useClipboard, useI18n } from '#imports'
export const useCopy = (showDialogIfFailed = false) => {
const { t } = useI18n()
export const useCopy = () => {
/** fallback for copy if clipboard api is not supported */ /** fallback for copy if clipboard api is not supported */
const copyFallback = (text: string) => { const copyFallback = async (text: string, retryCount = 0): Promise<boolean> => {
const textAreaEl = document.createElement('textarea') try {
textAreaEl.innerHTML = text const textAreaEl = document.createElement('textarea')
document.body.appendChild(textAreaEl) textAreaEl.innerHTML = text
textAreaEl.select() document.body.appendChild(textAreaEl)
const result = document.execCommand('copy') textAreaEl.select()
document.body.removeChild(textAreaEl) const result = document.execCommand('copy')
return result 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: _copy, isSupported } = useClipboard()
@ -18,12 +49,11 @@ export const useCopy = () => {
try { try {
if (isSupported.value) { if (isSupported.value) {
await _copy(text) await _copy(text)
} else { return true
copyFallback(text)
} }
} catch (e) { } catch {}
copyFallback(text)
} return copyFallback(text)
} }
return { copy } return { copy }

29
packages/nc-gui/pages/[projectType]/[projectId]/index.vue

@ -52,7 +52,7 @@ const { clearTabs, addTab } = useTabs()
const { isUIAllowed } = useUIPermission() const { isUIAllowed } = useUIPermission()
const { copy } = useCopy() const { copy } = useCopy(true)
// create a new sidebar state // create a new sidebar state
const { isOpen, toggle, toggleHasSidebar } = useSidebar('nc-left-sidebar', { hasSidebar: false, isOpen: false }) const { isOpen, toggle, toggleHasSidebar } = useSidebar('nc-left-sidebar', { hasSidebar: false, isOpen: false })
@ -128,15 +128,17 @@ const copyProjectInfo = async () => {
try { try {
await loadProjectMetaInfo() await loadProjectMetaInfo()
await copy( if (
Object.entries(projectMetaInfo.value!) await copy(
.map(([k, v]) => `${k}: **${v}**`) Object.entries(projectMetaInfo.value!)
.join('\n'), .map(([k, v]) => `${k}: **${v}**`)
) .join('\n'),
)
// Copied to clipboard ) {
message.info(t('msg.info.copiedToClipboard')) // Copied to clipboard
} catch (e: any) { message.info(t('msg.info.copiedToClipboard'))
}
} catch (e) {
console.error(e) console.error(e)
message.error(e.message) message.error(e.message)
} }
@ -144,9 +146,10 @@ const copyProjectInfo = async () => {
const copyAuthToken = async () => { const copyAuthToken = async () => {
try { try {
await copy(token.value!) if (await copy(token.value!)) {
// Copied to clipboard // Copied to clipboard
message.info(t('msg.info.copiedToClipboard')) message.info(t('msg.info.copiedToClipboard'))
}
} catch (e: any) { } catch (e: any) {
console.error(e) console.error(e)
message.error(e.message) message.error(e.message)

10
packages/nc-gui/pages/index/index/index.vue

@ -132,9 +132,13 @@ onBeforeMount(loadProjects)
const { copy } = useCopy() const { copy } = useCopy()
const copyProjectMeta = async () => { const copyProjectMeta = async () => {
const aggregatedMetaInfo = await $api.utils.aggregatedMetaInfo() try {
copy(JSON.stringify(aggregatedMetaInfo)) const aggregatedMetaInfo = await $api.utils.aggregatedMetaInfo()
message.info('Copied aggregated project meta to clipboard') await copy(JSON.stringify(aggregatedMetaInfo))
message.info('Copied aggregated project meta to clipboard')
} catch (e) {
message.error(await extractSdkResponseErrorMsg(e))
}
} }
</script> </script>

Loading…
Cancel
Save