Files
nocodb/packages/nc-gui/components/virtual-cell/barcode/JsBarcodeWrapper.vue
mertmit 69a29568c7 chore: sync
Signed-off-by: mertmit <mertmit99@gmail.com>
2026-01-10 00:21:02 +03:00

126 lines
3.8 KiB
Vue

<script lang="ts" setup>
import JsBarcode from 'jsbarcode'
import { downloadSvg as _downloadSvg, copySVGToClipboard } from '~/utils/svgToPng'
const props = defineProps({
barcodeValue: { type: String, required: true },
barcodeFormat: { type: String, required: true },
customStyle: { type: Object, required: false },
showDownload: { type: Boolean, required: false, default: false },
isModal: { type: Boolean, required: false, default: false },
})
const emit = defineEmits(['onClickBarcode'])
const { t } = useI18n()
const { isDark } = useTheme()
const isGallery = inject(IsGalleryInj, ref(false))
const barcodeSvgRef = ref<SVGGraphicsElement>()
const errorForCurrentInput = ref(false)
const generate = () => {
try {
JsBarcode(barcodeSvgRef.value, String(props.barcodeValue), {
format: props.barcodeFormat,
displayValue: false,
...(props.isModal && isDark.value
? { marginTop: 12, marginBottom: 12, marginLeft: 24, marginRight: 24 }
: isDark.value
? { marginTop: 4, marginBottom: 4, marginLeft: 8, marginRight: 8 }
: { margin: 0 }),
})
if (props.customStyle) {
if (barcodeSvgRef.value) {
for (const key in props.customStyle) {
barcodeSvgRef.value.style.setProperty(key, props.customStyle[key])
}
}
}
errorForCurrentInput.value = false
} catch (e) {
console.log('e', e)
errorForCurrentInput.value = true
}
}
const downloadSvg = () => {
if (!barcodeSvgRef.value) return
_downloadSvg(barcodeSvgRef.value, `${props.barcodeValue}.png`)
}
const { isCopied, performCopy } = useIsCopied()
const copyAsPng = async () => {
if (!barcodeSvgRef.value) return
const success = await copySVGToClipboard(barcodeSvgRef.value)
if (!success) throw new Error(t('msg.error.notSupported'))
}
const onBarcodeClick = (ev: MouseEvent) => {
ev.stopPropagation()
emit('onClickBarcode')
}
watch([() => props.barcodeValue, () => props.barcodeFormat, () => props.customStyle, () => isDark.value], generate)
onMounted(generate)
</script>
<template>
<div class="relative">
<svg
v-show="!errorForCurrentInput"
ref="barcodeSvgRef"
:class="{
'w-full': !isGallery,
'w-auto': isGallery,
'mt-8 mb-4': showDownload,
}"
data-testid="barcode"
@click="onBarcodeClick"
></svg>
<slot v-if="errorForCurrentInput" name="barcodeRenderError" />
<div v-if="showDownload" class="bg-nc-bg-gray-light mx-4 px-3 py-2 rounded-lg">
<NcTooltip show-on-truncate-only class="truncate">
<template #title>
{{ barcodeValue }}
</template>
{{ barcodeValue }}
</NcTooltip>
</div>
<div v-if="showDownload" class="flex justify-end gap-2 py-2 px-3">
<NcTooltip>
<template #title>
{{ $t('labels.clickToCopy') }}
</template>
<NcButton size="small" type="secondary" @click="performCopy(copyAsPng)">
<template #icon>
<div class="flex children:flex-none relative h-4 w-4">
<Transition name="icon-fade" :duration="200">
<GeneralIcon v-if="isCopied" icon="check" class="h-4 w-4 opacity-80" />
<GeneralIcon v-else icon="copy" class="h-4 w-4 opacity-80" />
</Transition>
</div>
</template>
{{ isCopied ? $t('general.copied') : $t('general.copy') }}
</NcButton>
</NcTooltip>
<NcTooltip>
<template #title>
{{ $t('labels.clickToDownload') }}
</template>
<NcButton size="small" type="secondary" @click="downloadSvg">
<template #icon>
<GeneralIcon icon="download" class="w-4 h-4" />
</template>
{{ $t('general.download') }}
</NcButton>
</NcTooltip>
</div>
</div>
</template>