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

148 lines
4.7 KiB
Vue

<script setup lang="ts">
import { useUploadState } from './useUploadState'
const { isMobileMode } = useGlobal()
const { tempFiles, addFiles, clearFiles, closeModal, isLoading } = useUploadState()
const videoStream = ref<MediaStream | null>(null)
const permissionGranted = ref(false)
const capturedImage = computed(() => tempFiles.value[0] || null)
const videoRef = ref<HTMLVideoElement | undefined>()
const canvasRef = ref<HTMLCanvasElement | undefined>()
const startCamera = async () => {
try {
if (!videoStream.value) {
videoStream.value = await navigator.mediaDevices.getUserMedia({ video: true })
}
permissionGranted.value = true
// Wait for next tick to ensure video element is rendered
await nextTick()
if (!videoRef.value || !videoStream.value) return
videoRef.value.srcObject = videoStream.value
try {
await videoRef.value.play()
} catch (playError) {
console.error('Error playing video:', playError)
}
} catch (error) {
console.error('Camera access denied:', error)
}
}
const stopCamera = () => {
videoStream.value?.getTracks().forEach((track) => track.stop())
videoStream.value = null
if (videoRef.value) {
videoRef.value.srcObject = null
}
}
const captureImage = () => {
const video = videoRef.value
const canvas = canvasRef.value
if (!video || !canvas) return
canvas.width = video.videoWidth
canvas.height = video.videoHeight
const context = canvas.getContext('2d')
if (context) {
canvas.style.display = 'block'
context.translate(canvas.width, 0)
context.scale(-1, 1)
context.drawImage(video, 0, 0, canvas.width, canvas.height)
canvas.toBlob((blob) => {
if (!blob) return
const file = new File([blob], `${new Date().toDateString()}.png`, { type: 'image/png' })
addFiles([file])
}, 'image/png')
stopCamera()
}
}
const retakeImage = () => {
clearFiles()
startCamera()
}
onMounted(() => {
startCamera()
})
onBeforeUnmount(() => {
stopCamera()
})
</script>
<template>
<div class="w-full relative h-full">
<NcTooltip class="absolute top-3 right-2 z-10">
<NcButton type="text" class="!border-0" :disabled="isLoading" :size="isMobileMode ? 'small' : 'xsmall'" @click="closeModal">
<GeneralIcon icon="close" />
</NcButton>
<template #title> {{ $t('general.close') }} </template>
</NcTooltip>
<div v-if="!permissionGranted" class="w-full h-full flex bg-nc-bg-gray-extralight items-center justify-center">
<div
class="flex flex-col hover:bg-nc-bg-default p-2 cursor-pointer rounded-md !transition-all transition-ease-in-out duration-300 gap-2 items-center justify-center"
@click="startCamera"
>
<div class="p-5 bg-nc-bg-default rounded-md shadow-sm">
<mdi-camera class="text-4xl text-nc-content-gray" />
</div>
<h1 class="text-nc-content-gray font-semibold text-center text-xl">
{{ $t('labels.allowAccessToYourCamera') }}
</h1>
</div>
</div>
<div
v-else
:class="{
'py-8': !capturedImage,
'pt-8 pb-2': capturedImage,
}"
class="w-full gap-3 h-full flex-col flex items-center justify-between"
>
<div v-show="!capturedImage" class="w-full gap-3 h-full flex-col flex items-center justify-between">
<video ref="videoRef" class="rounded-md w-full aspect-video max-w-md flex-1 object-contain" autoplay playsinline></video>
<NcButton class="!rounded-full !px-0" :disabled="isLoading" @click="captureImage">
<mdi-camera class="text-xl" />
</NcButton>
</div>
<div v-show="capturedImage" class="flex group flex-col">
<canvas ref="canvasRef" class="mb-2 rounded-md w-full aspect-video max-w-md flex-1 object-contain"></canvas>
<div class="relative text-[12px] font-semibold text-nc-content-gray flex">
<div class="flex-auto truncate line-height-4">
{{ capturedImage?.name }}
</div>
<div
v-if="!isLoading"
class="flex-none hide-ui transition-all transition-ease-in-out !h-4 flex items-center bg-nc-bg-default"
>
<NcTooltip placement="bottom">
<template #title> {{ $t('title.removeFile') }} </template>
<GeneralIcon icon="delete" class="!text-nc-content-red-medium cursor-pointer" @click="retakeImage" />
</NcTooltip>
</div>
</div>
<div class="flex-none text-[10px] font-semibold text-nc-content-gray-muted">
{{ formatBytes(capturedImage?.size, 0) }}
</div>
</div>
<NcFileUploadProvidersBottomBar v-show="capturedImage" class="pr-2 bottom-1 relative" upload-text="Upload Image" />
</div>
</div>
</template>