mirror of https://github.com/nocodb/nocodb
Wing-Kam Wong
2 years ago
21 changed files with 30008 additions and 16744 deletions
@ -0,0 +1,311 @@
|
||||
<script setup lang="ts"> |
||||
import io from 'socket.io-client' |
||||
import type { Socket } from 'socket.io-client' |
||||
import { Form } from 'ant-design-vue' |
||||
import { useToast } from 'vue-toastification' |
||||
import { fieldRequiredValidator } from '~/utils/validation' |
||||
import { extractSdkResponseErrorMsg } from '~/utils/errorUtils' |
||||
import MdiCloseCircleOutlineIcon from '~icons/mdi/close-circle-outline' |
||||
import MdiCurrencyUsdIcon from '~icons/mdi/currency-usd' |
||||
import MdiLoadingIcon from '~icons/mdi/loading' |
||||
|
||||
interface Props { |
||||
modelValue: boolean |
||||
} |
||||
|
||||
const { modelValue } = defineProps<Props>() |
||||
const emit = defineEmits(['update:modelValue']) |
||||
|
||||
// TODO: handle baseURL |
||||
const baseURL = 'http://localhost:8080' // this.$axios.defaults.baseURL |
||||
|
||||
const { $state } = useNuxtApp() |
||||
const toast = useToast() |
||||
const { sqlUi, project, loadTables } = useProject() |
||||
const loading = ref(false) |
||||
const step = ref(1) |
||||
const progress = ref<Record<string, any>[]>([]) |
||||
let socket: Socket | null |
||||
const syncSource = ref({ |
||||
id: '', |
||||
type: 'Airtable', |
||||
details: { |
||||
syncInterval: '15mins', |
||||
syncDirection: 'Airtable to NocoDB', |
||||
syncRetryCount: 1, |
||||
apiKey: '', |
||||
shareId: '', |
||||
syncSourceUrlOrId: '', |
||||
options: { |
||||
syncViews: true, |
||||
syncData: true, |
||||
syncRollup: false, |
||||
syncLookup: true, |
||||
syncFormula: false, |
||||
syncAttachment: true, |
||||
}, |
||||
}, |
||||
}) |
||||
|
||||
const validators = computed(() => { |
||||
return { |
||||
'details.apiKey': [fieldRequiredValidator], |
||||
'details.syncSourceUrlOrId': [fieldRequiredValidator], |
||||
} |
||||
}) |
||||
|
||||
const dialogShow = computed({ |
||||
get() { |
||||
return modelValue |
||||
}, |
||||
set(v) { |
||||
emit('update:modelValue', v) |
||||
}, |
||||
}) |
||||
|
||||
const useForm = Form.useForm |
||||
const { resetFields, validate, validateInfos } = useForm(syncSource, validators) |
||||
|
||||
const disableImportButton = computed(() => { |
||||
return !syncSource.value.details.apiKey || !syncSource.value.details.syncSourceUrlOrId |
||||
}) |
||||
|
||||
async function saveAndSync() { |
||||
await createOrUpdate() |
||||
await sync() |
||||
} |
||||
|
||||
async function createOrUpdate() { |
||||
try { |
||||
const { id, ...payload } = syncSource.value |
||||
if (id !== '') { |
||||
await $fetch(`/api/v1/db/meta/syncs/${id}`, { |
||||
baseURL, |
||||
method: 'PATCH', |
||||
headers: { 'xc-auth': $state.token.value as string }, |
||||
body: payload, |
||||
}) |
||||
} else { |
||||
const data: any = await $fetch(`/api/v1/db/meta/projects/${project.value.id}/syncs`, { |
||||
baseURL, |
||||
method: 'POST', |
||||
headers: { 'xc-auth': $state.token.value as string }, |
||||
body: payload, |
||||
}) |
||||
syncSource.value = data |
||||
} |
||||
} catch (e: any) { |
||||
toast.error(await extractSdkResponseErrorMsg(e)) |
||||
} |
||||
} |
||||
|
||||
async function loadSyncSrc() { |
||||
const data: any = await $fetch(`/api/v1/db/meta/projects/${project.value.id}/syncs`, { |
||||
baseURL, |
||||
method: 'GET', |
||||
headers: { 'xc-auth': $state.token.value as string }, |
||||
}) |
||||
const { list: srcs } = data |
||||
if (srcs && srcs[0]) { |
||||
srcs[0].details = srcs[0].details || {} |
||||
syncSource.value = migrateSync(srcs[0]) |
||||
syncSource.value.details.syncSourceUrlOrId = srcs[0].details.shareId |
||||
} else { |
||||
syncSource.value = { |
||||
id: '', |
||||
type: 'Airtable', |
||||
details: { |
||||
syncInterval: '15mins', |
||||
syncDirection: 'Airtable to NocoDB', |
||||
syncRetryCount: 1, |
||||
apiKey: '', |
||||
shareId: '', |
||||
syncSourceUrlOrId: '', |
||||
options: { |
||||
syncViews: true, |
||||
syncData: true, |
||||
syncRollup: false, |
||||
syncLookup: true, |
||||
syncFormula: false, |
||||
syncAttachment: true, |
||||
}, |
||||
}, |
||||
} |
||||
} |
||||
} |
||||
|
||||
async function sync() { |
||||
step.value = 2 |
||||
try { |
||||
await $fetch(`/api/v1/db/meta/syncs/${syncSource.value.id}/trigger`, { |
||||
baseURL, |
||||
method: 'POST', |
||||
headers: { 'xc-auth': $state.token.value as string }, |
||||
params: { |
||||
id: socket.id, |
||||
}, |
||||
}) |
||||
} catch (e: any) { |
||||
console.log(e) |
||||
toast.error(e) |
||||
} |
||||
} |
||||
|
||||
function migrateSync(src: any) { |
||||
if (!src.details?.options) { |
||||
src.details.options = { |
||||
syncViews: false, |
||||
syncData: true, |
||||
syncRollup: false, |
||||
syncLookup: true, |
||||
syncFormula: false, |
||||
syncAttachment: true, |
||||
} |
||||
src.details.options.syncViews = src.syncViews |
||||
delete src.syncViews |
||||
} |
||||
return src |
||||
} |
||||
|
||||
watch( |
||||
() => syncSource.value.details.syncSourceUrlOrId, |
||||
(v) => { |
||||
if (syncSource.value.details) { |
||||
const m = v && v.match(/(exp|shr).{14}/g) |
||||
syncSource.value.details.shareId = m ? m[0] : '' |
||||
} |
||||
}, |
||||
) |
||||
|
||||
onMounted(async () => { |
||||
socket = io(new URL(baseURL, window.location.href.split(/[?#]/)[0]).href, { |
||||
extraHeaders: { 'xc-auth': $state.token.value as string }, |
||||
}) |
||||
socket.on('connect_error', () => { |
||||
socket?.disconnect() |
||||
socket = null |
||||
}) |
||||
|
||||
// connect event does not provide data |
||||
socket.on('connect', () => { |
||||
console.log(socket?.id) |
||||
console.log('socket connected') |
||||
}) |
||||
|
||||
socket.on('progress', async (d: Record<string, any>) => { |
||||
progress.value.push(d) |
||||
if (d.status === 'COMPLETED') { |
||||
await loadTables() |
||||
// TODO: add tab of the first table |
||||
} |
||||
}) |
||||
await loadSyncSrc() |
||||
}) |
||||
|
||||
onBeforeUnmount(() => { |
||||
if (socket) { |
||||
socket.disconnect() |
||||
} |
||||
}) |
||||
</script> |
||||
|
||||
<template> |
||||
<a-modal v-model:visible="dialogShow" width="max(90vw, 600px)" @keydown.esc="dialogShow = false"> |
||||
<template #footer> |
||||
<div v-if="step === 1"> |
||||
<a-button key="back" @click="dialogShow = false">{{ $t('general.cancel') }}</a-button> |
||||
<a-button |
||||
key="submit" |
||||
v-t="['c:sync-airtable:save-and-sync']" |
||||
type="primary" |
||||
:disabled="disableImportButton" |
||||
@click="saveAndSync" |
||||
>Import |
||||
</a-button> |
||||
</div> |
||||
</template> |
||||
<a-typography-title class="ml-5 mt-5" type="secondary" :level="5">QUICK IMPORT - AIRTABLE</a-typography-title> |
||||
<div class="ml-5 mr-5"> |
||||
<a-divider /> |
||||
<div v-if="step === 1"> |
||||
<div class="mb-4"> |
||||
<span class="prose-xl font-bold mr-3">Credentials</span> |
||||
<a |
||||
href="https://docs.nocodb.com/setup-and-usages/import-airtable-to-sql-database-within-a-minute-for-free/#get-airtable-credentials" |
||||
class="prose-sm underline text-grey" |
||||
target="_blank" |
||||
>Where to find this? |
||||
</a> |
||||
</div> |
||||
<a-form ref="form" :model="syncSource" name="quick-import-airtable-form" layout="horizontal" class="ma-0"> |
||||
<a-form-item v-bind="validateInfos['details.apiKey']"> |
||||
<a-input-password v-model:value="syncSource.details.apiKey" placeholder="Api Key" size="large" /> |
||||
</a-form-item> |
||||
<a-form-item v-bind="validateInfos['details.syncSourceUrlOrId']"> |
||||
<a-input v-model:value="syncSource.details.syncSourceUrlOrId" placeholder="Shared Base ID / URL" size="large" /> |
||||
</a-form-item> |
||||
<span class="prose-xl font-bold self-center my-4">Advanced Settings</span> |
||||
<a-divider class="mt-2 mb-5" /> |
||||
<div class="mt-0 my-2"> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncData">Import Data</a-checkbox> |
||||
</div> |
||||
<div class="my-2"> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncViews">Import Secondary Views</a-checkbox> |
||||
</div> |
||||
<div class="my-2"> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncRollup">Import Rollup Columns</a-checkbox> |
||||
</div> |
||||
<div class="my-2"> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncLookup">Import Lookup Columns</a-checkbox> |
||||
</div> |
||||
<div class="my-2"> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncAttachment">Import Attachment Columns</a-checkbox> |
||||
</div> |
||||
<a-tooltip placement="top"> |
||||
<template #title> |
||||
<span>Coming Soon!</span> |
||||
</template> |
||||
<a-checkbox v-model:checked="syncSource.details.options.syncFormula" disabled>Import Formula Columns</a-checkbox> |
||||
</a-tooltip> |
||||
</a-form> |
||||
<a-divider /> |
||||
<div> |
||||
<a href="https://github.com/nocodb/nocodb/issues/2052" target="_blank">Questions / Help - Reach out here</a> |
||||
<br /> |
||||
<div> |
||||
This feature is currently in beta and more information can be found |
||||
<a class="prose-sm" href="https://github.com/nocodb/nocodb/discussions/2122" target="_blank">here</a>. |
||||
</div> |
||||
</div> |
||||
</div> |
||||
<div v-if="step === 2"> |
||||
<div class="mb-4 prose-xl font-bold">Logs</div> |
||||
<a-card body-style="background-color: #000000; height:400px; overflow: auto;"> |
||||
<div v-for="({ msg, status }, i) in progress" :key="i"> |
||||
<div v-if="status === 'FAILED'" class="flex items-center"> |
||||
<MdiCloseCircleOutlineIcon class="text-red-500" /> |
||||
<span class="text-red-500 ml-2">{{ msg }}</span> |
||||
</div> |
||||
<div v-else class="flex items-center"> |
||||
<MdiCurrencyUsdIcon class="text-green-500" /> |
||||
<span class="text-green-500 ml-2">{{ msg }}</span> |
||||
</div> |
||||
</div> |
||||
<div |
||||
v-if=" |
||||
!progress || |
||||
!progress.length || |
||||
(progress[progress.length - 1].status !== 'COMPLETED' && progress[progress.length - 1].status !== 'FAILED') |
||||
" |
||||
class="flex items-center" |
||||
> |
||||
<MdiLoadingIcon class="text-green-500 animate-spin" /> |
||||
<span class="text-green-500 ml-2"> Importing</span> |
||||
</div> |
||||
</a-card> |
||||
</div> |
||||
</div> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<style scoped lang="scss"></style> |
@ -0,0 +1,338 @@
|
||||
<script setup lang="ts"> |
||||
import { useToast } from 'vue-toastification' |
||||
import { Form } from 'ant-design-vue' |
||||
import type { TableType } from 'nocodb-sdk' |
||||
import type { UploadChangeParam } from 'ant-design-vue' |
||||
import { useI18n } from 'vue-i18n' |
||||
import MdiFileIcon from '~icons/mdi/file-plus-outline' |
||||
import MdiFileUploadOutlineIcon from '~icons/mdi/file-upload-outline' |
||||
import MdiLinkVariantIcon from '~icons/mdi/link-variant' |
||||
import MdiCodeJSONIcon from '~icons/mdi/code-json' |
||||
import { fieldRequiredValidator, importCsvUrlValidator, importExcelUrlValidator, importUrlValidator } from '~/utils/validation' |
||||
import { extractSdkResponseErrorMsg } from '~/utils/errorUtils' |
||||
import { ExcelTemplateAdapter, ExcelUrlTemplateAdapter, JSONTemplateAdapter, JSONUrlTemplateAdapter } from '~/utils/parsers' |
||||
import { useProject } from '#imports' |
||||
|
||||
interface Props { |
||||
modelValue: boolean |
||||
importType: 'csv' | 'json' | 'excel' |
||||
} |
||||
|
||||
const { modelValue, importType } = defineProps<Props>() |
||||
|
||||
const emit = defineEmits(['update:modelValue']) |
||||
|
||||
const { t } = useI18n() |
||||
|
||||
const { tables } = useProject() |
||||
|
||||
const toast = useToast() |
||||
|
||||
const activeKey = ref('uploadTab') |
||||
|
||||
const jsonEditorRef = ref() |
||||
|
||||
const templateEditorRef = ref() |
||||
|
||||
const loading = ref(false) |
||||
|
||||
const templateData = ref() |
||||
|
||||
const importData = ref() |
||||
|
||||
const templateEditorModal = ref(false) |
||||
|
||||
const useForm = Form.useForm |
||||
|
||||
const importState = reactive({ |
||||
fileList: [] as Record<string, any>, |
||||
url: '', |
||||
jsonEditor: {}, |
||||
parserConfig: { |
||||
maxRowsToParse: 500, |
||||
normalizeNested: true, |
||||
importData: true, |
||||
}, |
||||
}) |
||||
|
||||
const isImportTypeJson = computed(() => importType === 'json') |
||||
|
||||
const isImportTypeCsv = computed(() => importType === 'csv') |
||||
|
||||
const IsImportTypeExcel = computed(() => importType === 'excel') |
||||
|
||||
const validators = computed(() => { |
||||
return { |
||||
url: [fieldRequiredValidator, importUrlValidator, isImportTypeCsv.value ? importCsvUrlValidator : importExcelUrlValidator], |
||||
maxRowsToParse: [fieldRequiredValidator], |
||||
} |
||||
}) |
||||
|
||||
const { resetFields, validate, validateInfos } = useForm(importState, validators) |
||||
|
||||
const importMeta = computed(() => { |
||||
if (IsImportTypeExcel.value) { |
||||
return { |
||||
header: 'QUICK IMPORT - EXCEL', |
||||
uploadHint: t('msg.info.excelSupport'), |
||||
urlInputLabel: t('msg.info.excelURL'), |
||||
loadUrlDirective: ['c:quick-import:excel:load-url'], |
||||
acceptTypes: '.xls, .xlsx, .xlsm, .ods, .ots', |
||||
} |
||||
} else if (isImportTypeCsv.value) { |
||||
return { |
||||
header: 'QUICK IMPORT - CSV', |
||||
uploadHint: '', |
||||
urlInputLabel: t('msg.info.csvURL'), |
||||
loadUrlDirective: ['c:quick-import:csv:load-url'], |
||||
acceptTypes: '.csv', |
||||
} |
||||
} else if (isImportTypeJson.value) { |
||||
return { |
||||
header: 'QUICK IMPORT - JSON', |
||||
uploadHint: '', |
||||
acceptTypes: '.json', |
||||
} |
||||
} |
||||
return {} |
||||
}) |
||||
|
||||
const dialogShow = computed({ |
||||
get() { |
||||
return modelValue |
||||
}, |
||||
set(v) { |
||||
emit('update:modelValue', v) |
||||
}, |
||||
}) |
||||
|
||||
const disablePreImportButton = computed(() => { |
||||
if (activeKey.value === 'uploadTab') { |
||||
return !(importState.fileList.length > 0) |
||||
} else if (activeKey.value === 'urlTab') { |
||||
if (!validateInfos.url.validateStatus) return true |
||||
return validateInfos.url.validateStatus === 'error' |
||||
} else if (activeKey.value === 'jsonEditorTab') { |
||||
return !jsonEditorRef.value?.isValid |
||||
} |
||||
}) |
||||
|
||||
const disableImportButton = computed(() => { |
||||
return !templateEditorRef.value?.isValid |
||||
}) |
||||
|
||||
const disableFormatJsonButton = computed(() => !jsonEditorRef.value?.isValid) |
||||
|
||||
async function handlePreImport() { |
||||
loading.value = true |
||||
if (activeKey.value === 'uploadTab') { |
||||
await parseAndExtractData(importState.fileList[0].data, importState.fileList[0].name) |
||||
} else if (activeKey.value === 'urlTab') { |
||||
try { |
||||
await validate() |
||||
await parseAndExtractData(importState.url, '') |
||||
} catch (e: any) { |
||||
toast.error(await extractSdkResponseErrorMsg(e)) |
||||
} |
||||
} else if (activeKey.value === 'jsonEditorTab') { |
||||
await parseAndExtractData(JSON.stringify(importState.jsonEditor), '') |
||||
} |
||||
loading.value = false |
||||
} |
||||
|
||||
async function handleImport() { |
||||
loading.value = true |
||||
await templateEditorRef.value.importTemplate() |
||||
loading.value = false |
||||
dialogShow.value = false |
||||
} |
||||
|
||||
async function parseAndExtractData(val: any, name: string) { |
||||
try { |
||||
templateData.value = null |
||||
importData.value = null |
||||
const templateGenerator: any = getAdapter(name, val) |
||||
if (!templateGenerator) { |
||||
toast.error('Template Generator cannot be found!') |
||||
return |
||||
} |
||||
await templateGenerator.init() |
||||
templateGenerator.parse() |
||||
templateData.value = templateGenerator.getTemplate() |
||||
templateData.value.tables[0].table_name = populateUniqueTableName() |
||||
importData.value = templateGenerator.getData() |
||||
templateEditorModal.value = true |
||||
} catch (e: any) { |
||||
console.log(e) |
||||
toast.error(await extractSdkResponseErrorMsg(e)) |
||||
} |
||||
} |
||||
|
||||
function rejectDrop(fileList: any[]) { |
||||
fileList.map((file) => { |
||||
return toast.error(`Failed to upload file ${file.name}`) |
||||
}) |
||||
} |
||||
|
||||
function handleChange(info: UploadChangeParam) { |
||||
const status = info.file.status |
||||
if (status !== 'uploading') { |
||||
const reader: any = new FileReader() |
||||
reader.onload = (e: any) => { |
||||
const target: any = importState.fileList.find((f: any) => f?.uid === info.file.uid) |
||||
if (target) { |
||||
target.data = e.target.result |
||||
} |
||||
} |
||||
reader.readAsArrayBuffer(info.file.originFileObj) |
||||
} |
||||
if (status === 'done') { |
||||
toast.success(`Uploaded file ${info.file.name} successfully`) |
||||
} else if (status === 'error') { |
||||
toast.error(`Failed to upload file ${info.file.name}`) |
||||
} |
||||
} |
||||
|
||||
function formatJson() { |
||||
jsonEditorRef.value?.format() |
||||
} |
||||
|
||||
function populateUniqueTableName() { |
||||
let c = 1 |
||||
while (tables.value.some((t: TableType) => t.title === `Sheet${c}`)) { |
||||
c++ |
||||
} |
||||
return `Sheet${c}` |
||||
} |
||||
|
||||
function getAdapter(name: string, val: any) { |
||||
if (IsImportTypeExcel.value || isImportTypeCsv.value) { |
||||
switch (activeKey.value) { |
||||
case 'uploadTab': |
||||
return new ExcelTemplateAdapter(name, val, importState.parserConfig) |
||||
case 'urlTab': |
||||
return new ExcelUrlTemplateAdapter(val, importState.parserConfig) |
||||
} |
||||
} else if (isImportTypeJson.value) { |
||||
switch (activeKey.value) { |
||||
case 'uploadTab': |
||||
return new JSONTemplateAdapter(name, val, importState.parserConfig) |
||||
case 'urlTab': |
||||
return new JSONUrlTemplateAdapter(val, importState.parserConfig) |
||||
case 'jsonEditorTab': |
||||
return new JSONTemplateAdapter(name, val, importState.parserConfig) |
||||
} |
||||
} |
||||
return null |
||||
} |
||||
</script> |
||||
|
||||
<template> |
||||
<a-modal v-model:visible="dialogShow" width="max(90vw, 600px)" @keydown.esc="dialogShow = false"> |
||||
<a-typography-title class="ml-5 mt-5 mb-5" type="secondary" :level="5">{{ importMeta.header }}</a-typography-title> |
||||
<template #footer> |
||||
<a-button v-if="templateEditorModal" key="back" @click="templateEditorModal = false">Back</a-button> |
||||
<a-button v-else key="cancel" @click="dialogShow = false">{{ $t('general.cancel') }}</a-button> |
||||
<a-button v-if="activeKey === 'jsonEditorTab'" key="format" :disabled="disableFormatJsonButton" @click="formatJson" |
||||
>Format JSON</a-button |
||||
> |
||||
<a-button |
||||
v-if="!templateEditorModal" |
||||
key="pre-import" |
||||
type="primary" |
||||
:loading="loading" |
||||
:disabled="disablePreImportButton" |
||||
@click="handlePreImport" |
||||
>{{ $t('activity.import') }} |
||||
</a-button> |
||||
<a-button v-else key="import" type="primary" :loading="loading" :disabled="disableImportButton" @click="handleImport">{{ |
||||
$t('activity.import') |
||||
}}</a-button> |
||||
</template> |
||||
<div class="ml-5 mr-5"> |
||||
<TemplateEditor |
||||
v-if="templateEditorModal" |
||||
ref="templateEditorRef" |
||||
:project-template="templateData" |
||||
:import-data="importData" |
||||
:quick-import-type="importType" |
||||
/> |
||||
<a-tabs v-else v-model:activeKey="activeKey" hide-add type="editable-card" :tab-position="top"> |
||||
<a-tab-pane key="uploadTab" :closable="false"> |
||||
<template #tab> |
||||
<span class="flex items-center gap-2"> |
||||
<MdiFileUploadOutlineIcon /> |
||||
Upload |
||||
</span> |
||||
</template> |
||||
<div class="pr-10 pb-0 pt-5"> |
||||
<a-upload-dragger |
||||
v-model:fileList="importState.fileList" |
||||
name="file" |
||||
:accept="importMeta.acceptTypes" |
||||
:max-count="1" |
||||
list-type="picture" |
||||
@change="handleChange" |
||||
@reject="rejectDrop" |
||||
> |
||||
<MdiFileIcon size="large" /> |
||||
<p class="ant-upload-text">Click or drag file to this area to upload</p> |
||||
<p class="ant-upload-hint"> |
||||
{{ importMeta.uploadHint }} |
||||
</p> |
||||
</a-upload-dragger> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane v-if="isImportTypeJson" key="jsonEditorTab" :closable="false"> |
||||
<template #tab> |
||||
<span class="flex items-center gap-2"> |
||||
<MdiCodeJSONIcon /> |
||||
Json Editor |
||||
</span> |
||||
</template> |
||||
<div class="pb-3 pt-3"> |
||||
<MonacoEditor ref="jsonEditorRef" v-model="importState.jsonEditor" class="min-h-60 max-h-80" /> |
||||
</div> |
||||
</a-tab-pane> |
||||
<a-tab-pane v-else key="urlTab" :closable="false"> |
||||
<template #tab> |
||||
<span class="flex items-center gap-2"> |
||||
<MdiLinkVariantIcon /> |
||||
Url |
||||
</span> |
||||
</template> |
||||
<div class="pr-10 pt-5"> |
||||
<a-form :model="importState" name="quick-import-url-form" layout="horizontal" class="mb-0"> |
||||
<a-form-item :label="importMeta.urlInputLabel" v-bind="validateInfos.url"> |
||||
<a-input v-model:value="importState.url" size="large" /> |
||||
</a-form-item> |
||||
</a-form> |
||||
</div> |
||||
</a-tab-pane> |
||||
</a-tabs> |
||||
</div> |
||||
<div v-if="!templateEditorModal" class="ml-5 mr-5"> |
||||
<a-divider /> |
||||
<div class="mb-4"> |
||||
<span class="prose-xl font-bold">Advanced Settings</span> |
||||
<a-form-item class="mt-4 mb-2" :label="t('msg.info.footMsg')" v-bind="validateInfos.maxRowsToParse"> |
||||
<a-input-number v-model:value="importState.parserConfig.maxRowsToParse" :min="1" :max="50000" /> |
||||
</a-form-item> |
||||
<div v-if="isImportTypeJson" class="mt-3"> |
||||
<a-checkbox v-model:checked="importState.parserConfig.normalizeNested"> |
||||
<span class="caption">Flatten nested</span> |
||||
</a-checkbox> |
||||
</div> |
||||
<div v-if="isImportTypeJson" class="mt-4"> |
||||
<a-checkbox v-model:checked="importState.parserConfig.importData">Import data</a-checkbox> |
||||
</div> |
||||
</div> |
||||
</div> |
||||
</a-modal> |
||||
</template> |
||||
|
||||
<style scoped lang="scss"> |
||||
:deep(.ant-upload-list) { |
||||
@apply max-h-80 overflow-auto; |
||||
} |
||||
</style> |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,42 @@
|
||||
import Papaparse from 'papaparse' |
||||
import TemplateGenerator from './TemplateGenerator' |
||||
|
||||
export default class CSVTemplateAdapter extends TemplateGenerator { |
||||
fileName: string |
||||
project: object |
||||
data: object |
||||
csv: any |
||||
csvData: any |
||||
columns: object |
||||
|
||||
constructor(name: string, data: object) { |
||||
super() |
||||
this.fileName = name |
||||
this.csvData = data |
||||
this.project = { |
||||
title: this.fileName, |
||||
tables: [], |
||||
} |
||||
this.data = {} |
||||
this.csv = {} |
||||
this.columns = {} |
||||
this.csvData = {} |
||||
} |
||||
|
||||
async init() { |
||||
this.csv = Papaparse.parse(this.csvData, { header: true }) |
||||
} |
||||
|
||||
parseData() { |
||||
this.columns = this.csv.meta.fields |
||||
this.data = this.csv.data |
||||
} |
||||
|
||||
getColumns() { |
||||
return this.columns |
||||
} |
||||
|
||||
getData() { |
||||
return this.data |
||||
} |
||||
} |
@ -0,0 +1,240 @@
|
||||
import XLSX from 'xlsx' |
||||
import { UITypes } from 'nocodb-sdk' |
||||
import TemplateGenerator from './TemplateGenerator' |
||||
import { getCheckboxValue, isCheckboxType } from './parserHelpers' |
||||
|
||||
const excelTypeToUidt: Record<any, any> = { |
||||
d: UITypes.DateTime, |
||||
b: UITypes.Checkbox, |
||||
n: UITypes.Number, |
||||
s: UITypes.SingleLineText, |
||||
} |
||||
|
||||
export default class ExcelTemplateAdapter extends TemplateGenerator { |
||||
config: Record<string, any> |
||||
name: string |
||||
excelData: any |
||||
project: Record<string, any> |
||||
data: Record<string, any> |
||||
wb: any |
||||
constructor(name = '', data = {}, parserConfig = {}) { |
||||
super() |
||||
this.config = { |
||||
maxRowsToParse: 500, |
||||
...parserConfig, |
||||
} |
||||
this.name = name |
||||
this.excelData = data |
||||
this.project = { |
||||
title: this.name, |
||||
tables: [], |
||||
} |
||||
this.data = {} |
||||
} |
||||
|
||||
async init() { |
||||
const options: Record<any, boolean> = { |
||||
cellText: true, |
||||
cellDates: true, |
||||
} |
||||
if (this.name.slice(-3) === 'csv') { |
||||
this.wb = XLSX.read(new TextDecoder().decode(new Uint8Array(this.excelData)), { |
||||
type: 'string', |
||||
...options, |
||||
}) |
||||
} else { |
||||
this.wb = XLSX.read(new Uint8Array(this.excelData), { |
||||
type: 'array', |
||||
...options, |
||||
}) |
||||
} |
||||
} |
||||
|
||||
parse() { |
||||
const tableNamePrefixRef: any = {} |
||||
for (let i = 0; i < this.wb.SheetNames.length; i++) { |
||||
const columnNamePrefixRef: Record<any, any> = { id: 0 } |
||||
const sheet: any = this.wb.SheetNames[i] |
||||
let tn: string = (sheet || 'table').replace(/[` ~!@#$%^&*()_|+\-=?;:'",.<>\{\}\[\]\\\/]/g, '_').trim() |
||||
|
||||
while (tn in tableNamePrefixRef) { |
||||
tn = `${tn}${++tableNamePrefixRef[tn]}` |
||||
} |
||||
tableNamePrefixRef[tn] = 0 |
||||
|
||||
const table: Record<string, any> = { table_name: tn, ref_table_name: tn, columns: [] } |
||||
this.data[tn] = [] |
||||
const ws: any = this.wb.Sheets[sheet] |
||||
const range = XLSX.utils.decode_range(ws['!ref']) |
||||
let rows: any = XLSX.utils.sheet_to_json(ws, { header: 1, blankrows: false, defval: null }) |
||||
|
||||
if (this.name.slice(-3) !== 'csv') { |
||||
// fix precision bug & timezone offset issues introduced by xlsx
|
||||
const basedate = new Date(1899, 11, 30, 0, 0, 0) |
||||
// number of milliseconds since base date
|
||||
const dnthresh = basedate.getTime() + (new Date().getTimezoneOffset() - basedate.getTimezoneOffset()) * 60000 |
||||
// number of milliseconds in a day
|
||||
const day_ms = 24 * 60 * 60 * 1000 |
||||
// handle date1904 property
|
||||
const fixImportedDate = (date: Date) => { |
||||
const parsed = XLSX.SSF.parse_date_code((date.getTime() - dnthresh) / day_ms, { |
||||
date1904: this.wb.Workbook.WBProps.date1904, |
||||
}) |
||||
return new Date(parsed.y, parsed.m, parsed.d, parsed.H, parsed.M, parsed.S) |
||||
} |
||||
// fix imported date
|
||||
rows = rows.map((r: any) => |
||||
r.map((v: any) => { |
||||
return v instanceof Date ? fixImportedDate(v) : v |
||||
}), |
||||
) |
||||
} |
||||
|
||||
const columnNameRowExist = +rows[0].every((v: any) => v === null || typeof v === 'string') |
||||
|
||||
// const colLen = Math.max()
|
||||
for (let col = 0; col < rows[0].length; col++) { |
||||
let cn: string = ((columnNameRowExist && rows[0] && rows[0][col] && rows[0][col].toString().trim()) || `field_${col + 1}`) |
||||
.replace(/[` ~!@#$%^&*()_|+\-=?;:'",.<>\{\}\[\]\\\/]/g, '_') |
||||
.trim() |
||||
|
||||
while (cn in columnNamePrefixRef) { |
||||
cn = `${cn}${++columnNamePrefixRef[cn]}` |
||||
} |
||||
columnNamePrefixRef[cn] = 0 |
||||
|
||||
const column: Record<string, any> = { |
||||
column_name: cn, |
||||
ref_column_name: cn, |
||||
} |
||||
|
||||
table.columns.push(column) |
||||
|
||||
// const cellId = `${col.toString(26).split('').map(s => (parseInt(s, 26) + 10).toString(36).toUpperCase())}2`;
|
||||
const cellId = XLSX.utils.encode_cell({ |
||||
c: range.s.c + col, |
||||
r: columnNameRowExist, |
||||
}) |
||||
const cellProps = ws[cellId] || {} |
||||
column.uidt = excelTypeToUidt[cellProps.t] || UITypes.SingleLineText |
||||
|
||||
// todo: optimize
|
||||
if (column.uidt === UITypes.SingleLineText) { |
||||
// check for long text
|
||||
if (rows.some((r: any) => (r[col] || '').toString().match(/[\r\n]/) || (r[col] || '').toString().length > 255)) { |
||||
column.uidt = UITypes.LongText |
||||
} else { |
||||
const vals = rows |
||||
.slice(columnNameRowExist ? 1 : 0) |
||||
.map((r: any) => r[col]) |
||||
.filter((v: any) => v !== null && v !== undefined && v.toString().trim() !== '') |
||||
|
||||
const checkboxType = isCheckboxType(vals) |
||||
if (checkboxType.length === 1) { |
||||
column.uidt = UITypes.Checkbox |
||||
} else { |
||||
// todo: optimize
|
||||
// check column is multi or single select by comparing unique values
|
||||
// todo:
|
||||
if (vals.some((v: any) => v && v.toString().includes(','))) { |
||||
let flattenedVals = vals.flatMap((v: any) => |
||||
v |
||||
? v |
||||
.toString() |
||||
.trim() |
||||
.split(/\s*,\s*/) |
||||
: [], |
||||
) |
||||
const uniqueVals = (flattenedVals = flattenedVals.filter( |
||||
(v: any, i: any, arr: any) => i === arr.findIndex((v1: any) => v.toLowerCase() === v1.toLowerCase()), |
||||
)) |
||||
if (flattenedVals.length > uniqueVals.length && uniqueVals.length <= Math.ceil(flattenedVals.length / 2)) { |
||||
column.uidt = UITypes.MultiSelect |
||||
column.dtxp = `'${uniqueVals.join("','")}'` |
||||
} |
||||
} else { |
||||
const uniqueVals = vals |
||||
.map((v: any) => v.toString().trim()) |
||||
.filter((v: any, i: any, arr: any) => i === arr.findIndex((v1: any) => v.toLowerCase() === v1.toLowerCase())) |
||||
if (vals.length > uniqueVals.length && uniqueVals.length <= Math.ceil(vals.length / 2)) { |
||||
column.uidt = UITypes.SingleSelect |
||||
column.dtxp = `'${uniqueVals.join("','")}'` |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} else if (column.uidt === UITypes.Number) { |
||||
if ( |
||||
rows.slice(1, this.config.maxRowsToParse).some((v: any) => { |
||||
return v && v[col] && parseInt(v[col]) !== +v[col] |
||||
}) |
||||
) { |
||||
column.uidt = UITypes.Decimal |
||||
} |
||||
if ( |
||||
rows.slice(1, this.config.maxRowsToParse).every((v: any, i: any) => { |
||||
const cellId = XLSX.utils.encode_cell({ |
||||
c: range.s.c + col, |
||||
r: i + columnNameRowExist, |
||||
}) |
||||
|
||||
const cellObj = ws[cellId] |
||||
|
||||
return !cellObj || (cellObj.w && cellObj.w.startsWith('$')) |
||||
}) |
||||
) { |
||||
column.uidt = UITypes.Currency |
||||
} |
||||
} else if (column.uidt === UITypes.DateTime) { |
||||
if ( |
||||
rows.slice(1, this.config.maxRowsToParse).every((v: any, i: any) => { |
||||
const cellId = XLSX.utils.encode_cell({ |
||||
c: range.s.c + col, |
||||
r: i + columnNameRowExist, |
||||
}) |
||||
|
||||
const cellObj = ws[cellId] |
||||
return !cellObj || (cellObj.w && cellObj.w.split(' ').length === 1) |
||||
}) |
||||
) { |
||||
column.uidt = UITypes.Date |
||||
} |
||||
} |
||||
} |
||||
|
||||
let rowIndex = 0 |
||||
for (const row of rows.slice(1)) { |
||||
const rowData: Record<string, any> = {} |
||||
for (let i = 0; i < table.columns.length; i++) { |
||||
if (table.columns[i].uidt === UITypes.Checkbox) { |
||||
rowData[table.columns[i].column_name] = getCheckboxValue(row[i]) |
||||
} else if (table.columns[i].uidt === UITypes.Currency) { |
||||
const cellId = XLSX.utils.encode_cell({ |
||||
c: range.s.c + i, |
||||
r: rowIndex + columnNameRowExist, |
||||
}) |
||||
|
||||
const cellObj = ws[cellId] |
||||
rowData[table.columns[i].column_name] = (cellObj && cellObj.w && cellObj.w.replace(/[^\d.]+/g, '')) || row[i] |
||||
} else if (table.columns[i].uidt === UITypes.SingleSelect || table.columns[i].uidt === UITypes.MultiSelect) { |
||||
rowData[table.columns[i].column_name] = (row[i] || '').toString().trim() || null |
||||
} else { |
||||
// toto: do parsing if necessary based on type
|
||||
rowData[table.columns[i].column_name] = row[i] |
||||
} |
||||
} |
||||
this.data[tn].push(rowData) |
||||
rowIndex++ |
||||
} |
||||
this.project.tables.push(table) |
||||
} |
||||
} |
||||
|
||||
getTemplate() { |
||||
return this.project |
||||
} |
||||
|
||||
getData() { |
||||
return this.data |
||||
} |
||||
} |
@ -0,0 +1,27 @@
|
||||
import ExcelTemplateAdapter from './ExcelTemplateAdapter' |
||||
import { useNuxtApp } from '#imports' |
||||
|
||||
export default class ExcelUrlTemplateAdapter extends ExcelTemplateAdapter { |
||||
url: string |
||||
excelData: any |
||||
$api: any |
||||
|
||||
constructor(url: string, parserConfig: Record<string, any>) { |
||||
const { $api } = useNuxtApp() |
||||
const name = url?.split('/').pop() |
||||
super(name, parserConfig) |
||||
this.url = url |
||||
this.excelData = null |
||||
this.$api = $api |
||||
} |
||||
|
||||
async init() { |
||||
const data: any = await this.$api.utils.axiosRequestMake({ |
||||
apiMeta: { |
||||
url: this.url, |
||||
}, |
||||
}) |
||||
this.excelData = data.data |
||||
await super.init() |
||||
} |
||||
} |
@ -0,0 +1,168 @@
|
||||
import { UITypes } from 'nocodb-sdk' |
||||
import { |
||||
extractMultiOrSingleSelectProps, |
||||
getCheckboxValue, |
||||
isCheckboxType, |
||||
isDecimalType, |
||||
isEmailType, |
||||
isMultiLineTextType, |
||||
isUrlType, |
||||
} from './parserHelpers' |
||||
import TemplateGenerator from './TemplateGenerator' |
||||
|
||||
const jsonTypeToUidt: Record<string, string> = { |
||||
number: UITypes.Number, |
||||
string: UITypes.SingleLineText, |
||||
date: UITypes.DateTime, |
||||
boolean: UITypes.Checkbox, |
||||
object: UITypes.JSON, |
||||
} |
||||
|
||||
const extractNestedData: any = (obj: any, path: any) => path.reduce((val: any, key: any) => val && val[key], obj) |
||||
|
||||
export default class JSONTemplateAdapter extends TemplateGenerator { |
||||
config: Record<string, any> |
||||
name: string |
||||
data: Record<string, any> |
||||
_jsonData: string | Record<string, any> |
||||
jsonData: Record<string, any> |
||||
project: Record<string, any> |
||||
columns: object |
||||
constructor(name = 'test', data: object, parserConfig = {}) { |
||||
super() |
||||
this.config = { |
||||
maxRowsToParse: 500, |
||||
...parserConfig, |
||||
} |
||||
this.name = name |
||||
this._jsonData = data |
||||
this.project = { |
||||
title: this.name, |
||||
tables: [], |
||||
} |
||||
this.jsonData = [] |
||||
this.data = [] |
||||
this.columns = {} |
||||
} |
||||
|
||||
async init() { |
||||
const parsedJsonData = |
||||
typeof this._jsonData === 'string' |
||||
? // for json editor
|
||||
JSON.parse(this._jsonData) |
||||
: // for file upload
|
||||
JSON.parse(new TextDecoder().decode(this._jsonData as BufferSource)) |
||||
this.jsonData = Array.isArray(parsedJsonData) ? parsedJsonData : [parsedJsonData] |
||||
} |
||||
|
||||
getColumns(): any { |
||||
return this.columns |
||||
} |
||||
|
||||
getData(): any { |
||||
return this.data |
||||
} |
||||
|
||||
parse(): any { |
||||
const jsonData = this.jsonData |
||||
const tn = 'table' |
||||
const table: any = { table_name: tn, ref_table_name: tn, columns: [] } |
||||
|
||||
this.data[tn] = [] |
||||
|
||||
for (const col of Object.keys(jsonData[0])) { |
||||
const columns = this._parseColumn([col], jsonData) |
||||
table.columns.push(...columns) |
||||
} |
||||
|
||||
if (this.config.importData) { |
||||
this._parseTableData(table) |
||||
} |
||||
|
||||
this.project.tables.push(table) |
||||
} |
||||
|
||||
getTemplate() { |
||||
return this.project |
||||
} |
||||
|
||||
_parseColumn( |
||||
path: any = [], |
||||
jsonData = this.jsonData, |
||||
firstRowVal = path.reduce((val: any, k: any) => val && val[k], this.jsonData[0]), |
||||
): any { |
||||
const columns = [] |
||||
// parse nested
|
||||
if (firstRowVal && typeof firstRowVal === 'object' && !Array.isArray(firstRowVal) && this.config.normalizeNested) { |
||||
for (const key of Object.keys(firstRowVal)) { |
||||
const normalizedNestedColumns = this._parseColumn([...path, key], this.jsonData, firstRowVal[key]) |
||||
columns.push(...normalizedNestedColumns) |
||||
} |
||||
} else { |
||||
const cn = path.join('_').replace(/\W/g, '_').trim() |
||||
|
||||
const column: Record<string, any> = { |
||||
column_name: cn, |
||||
ref_column_name: cn, |
||||
path, |
||||
} |
||||
|
||||
column.uidt = jsonTypeToUidt[typeof firstRowVal] || UITypes.SingleLineText |
||||
|
||||
const colData = jsonData.map((r: any) => extractNestedData(r, path)) |
||||
Object.assign(column, this._getColumnUIDTAndMetas(colData, column.uidt)) |
||||
columns.push(column) |
||||
} |
||||
|
||||
return columns |
||||
} |
||||
|
||||
_getColumnUIDTAndMetas(colData: any, defaultType: any) { |
||||
const colProps = { uidt: defaultType } |
||||
// todo: optimize
|
||||
if (colProps.uidt === UITypes.SingleLineText) { |
||||
// check for long text
|
||||
if (isMultiLineTextType(colData)) { |
||||
colProps.uidt = UITypes.LongText |
||||
} |
||||
if (isEmailType(colData)) { |
||||
colProps.uidt = UITypes.Email |
||||
} |
||||
if (isUrlType(colData)) { |
||||
colProps.uidt = UITypes.URL |
||||
} else { |
||||
const checkboxType = isCheckboxType(colData) |
||||
if (checkboxType.length === 1) { |
||||
colProps.uidt = UITypes.Checkbox |
||||
} else { |
||||
Object.assign(colProps, extractMultiOrSingleSelectProps(colData)) |
||||
} |
||||
} |
||||
} else if (colProps.uidt === UITypes.Number) { |
||||
if (isDecimalType(colData)) { |
||||
colProps.uidt = UITypes.Decimal |
||||
} |
||||
} |
||||
return colProps |
||||
} |
||||
|
||||
_parseTableData(tableMeta: any) { |
||||
for (const row of this.jsonData as any) { |
||||
const rowData: any = {} |
||||
for (let i = 0; i < tableMeta.columns.length; i++) { |
||||
const value = extractNestedData(row, tableMeta.columns[i].path || []) |
||||
if (tableMeta.columns[i].uidt === UITypes.Checkbox) { |
||||
rowData[tableMeta.columns[i].ref_column_name] = getCheckboxValue(value) |
||||
} else if (tableMeta.columns[i].uidt === UITypes.SingleSelect || tableMeta.columns[i].uidt === UITypes.MultiSelect) { |
||||
rowData[tableMeta.columns[i].ref_column_name] = (value || '').toString().trim() || null |
||||
} else if (tableMeta.columns[i].uidt === UITypes.JSON) { |
||||
rowData[tableMeta.columns[i].ref_column_name] = JSON.stringify(value) |
||||
} else { |
||||
// toto: do parsing if necessary based on type
|
||||
rowData[tableMeta.columns[i].column_name] = value |
||||
} |
||||
} |
||||
this.data[tableMeta.ref_table_name].push(rowData) |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,25 @@
|
||||
import JSONTemplateAdapter from './JSONTemplateAdapter' |
||||
import { useNuxtApp } from '#imports' |
||||
|
||||
export default class JSONUrlTemplateAdapter extends JSONTemplateAdapter { |
||||
url: string |
||||
$api: any |
||||
|
||||
constructor(url: string, parserConfig: Record<string, any>) { |
||||
const { $api } = useNuxtApp() |
||||
const name = url.split('/').pop() |
||||
super(name, parserConfig) |
||||
this.url = url |
||||
this.$api = $api |
||||
} |
||||
|
||||
async init() { |
||||
const data = await this.$api.utils.axiosRequestMake({ |
||||
apiMeta: { |
||||
url: this.url, |
||||
}, |
||||
}) |
||||
this._jsonData = data |
||||
await super.init() |
||||
} |
||||
} |
@ -0,0 +1,25 @@
|
||||
export default class TemplateGenerator { |
||||
parse() { |
||||
throw new Error("'parse' method is not implemented") |
||||
} |
||||
|
||||
parseData() { |
||||
throw new Error("'parseData' method is not implemented") |
||||
} |
||||
|
||||
parseTemplate() { |
||||
throw new Error("'parseTemplate' method is not implemented") |
||||
} |
||||
|
||||
getColumns() { |
||||
throw new Error("'getColumns' method is not implemented") |
||||
} |
||||
|
||||
getTemplate() { |
||||
throw new Error("'getTemplate' method is not implemented") |
||||
} |
||||
|
||||
getData() { |
||||
throw new Error("'getData' method is not implemented") |
||||
} |
||||
} |
@ -0,0 +1,7 @@
|
||||
import JSONTemplateAdapter from './JSONTemplateAdapter' |
||||
import JSONUrlTemplateAdapter from './JSONUrlTemplateAdapter' |
||||
import CSVTemplateAdapter from './CSVTemplateAdapter' |
||||
import ExcelTemplateAdapter from './ExcelTemplateAdapter' |
||||
import ExcelUrlTemplateAdapter from './ExcelUrlTemplateAdapter' |
||||
|
||||
export { JSONTemplateAdapter, JSONUrlTemplateAdapter, CSVTemplateAdapter, ExcelTemplateAdapter, ExcelUrlTemplateAdapter } |
@ -0,0 +1,95 @@
|
||||
import { UITypes } from 'nocodb-sdk' |
||||
import { isValidURL } from '~/utils/urlUtils' |
||||
import { isEmail } from '~/utils/validation' |
||||
|
||||
const booleanOptions = [ |
||||
{ checked: true, unchecked: false }, |
||||
{ 'x': true, '': false }, |
||||
{ yes: true, no: false }, |
||||
{ y: true, n: false }, |
||||
{ 1: true, 0: false }, |
||||
{ '[x]': true, '[]': false, '[ ]': false }, |
||||
{ '☑': true, '': false }, |
||||
{ '✅': true, '': false }, |
||||
{ '✓': true, '': false }, |
||||
{ '✔': true, '': false }, |
||||
{ enabled: true, disabled: false }, |
||||
{ on: true, off: false }, |
||||
{ 'done': true, '': false }, |
||||
{ true: true, false: false }, |
||||
] |
||||
const aggBooleanOptions: any = booleanOptions.reduce((obj, o) => ({ ...obj, ...o }), {}) |
||||
|
||||
const getColVal = (row: any, col = null) => { |
||||
return row && col ? row[col] : row |
||||
} |
||||
|
||||
export const isCheckboxType: any = (values: [], col = null) => { |
||||
let options = booleanOptions |
||||
for (let i = 0; i < values.length; i++) { |
||||
const val = getColVal(values[i], col) |
||||
|
||||
if (val === null || val === undefined || val.toString().trim() === '') { |
||||
continue |
||||
} |
||||
|
||||
options = options.filter((v) => val in v) |
||||
if (!options.length) { |
||||
return false |
||||
} |
||||
} |
||||
return options |
||||
} |
||||
export const getCheckboxValue = (value: number) => { |
||||
return value && aggBooleanOptions[value] |
||||
} |
||||
|
||||
export const isMultiLineTextType = (values: [], col = null) => { |
||||
return values.some( |
||||
(r) => (getColVal(r, col) || '').toString().match(/[\r\n]/) || (getColVal(r, col) || '').toString().length > 255, |
||||
) |
||||
} |
||||
|
||||
export const extractMultiOrSingleSelectProps = (colData: []) => { |
||||
const colProps: any = {} |
||||
if (colData.some((v: any) => v && (v || '').toString().includes(','))) { |
||||
let flattenedVals = colData.flatMap((v: any) => |
||||
v |
||||
? v |
||||
.toString() |
||||
.trim() |
||||
.split(/\s*,\s*/) |
||||
: [], |
||||
) |
||||
const uniqueVals = (flattenedVals = flattenedVals.filter( |
||||
(v, i, arr) => i === arr.findIndex((v1) => v.toLowerCase() === v1.toLowerCase()), |
||||
)) |
||||
if (flattenedVals.length > uniqueVals.length && uniqueVals.length <= Math.ceil(flattenedVals.length / 2)) { |
||||
colProps.uidt = UITypes.MultiSelect |
||||
colProps.dtxp = `'${uniqueVals.join("','")}'` |
||||
} |
||||
} else { |
||||
const uniqueVals = colData |
||||
.map((v: any) => (v || '').toString().trim()) |
||||
.filter((v, i, arr) => i === arr.findIndex((v1) => v.toLowerCase() === v1.toLowerCase())) |
||||
if (colData.length > uniqueVals.length && uniqueVals.length <= Math.ceil(colData.length / 2)) { |
||||
colProps.uidt = UITypes.SingleSelect |
||||
colProps.dtxp = `'${uniqueVals.join("','")}'` |
||||
} |
||||
} |
||||
return colProps |
||||
} |
||||
|
||||
export const isDecimalType = (colData: []) => |
||||
colData.some((v: any) => { |
||||
return v && parseInt(v) !== +v |
||||
}) |
||||
|
||||
export const isEmailType = (colData: []) => |
||||
!colData.some((v: any) => { |
||||
return v && !isEmail(v) |
||||
}) |
||||
export const isUrlType = (colData: []) => |
||||
!colData.some((v: any) => { |
||||
return v && !isValidURL(v) |
||||
}) |
Loading…
Reference in new issue