2023-02-02 13:42:09 +05:30
|
|
|
<template>
|
|
|
|
<div class="flex flex-col overflow-hidden w-full">
|
|
|
|
<!-- Header -->
|
|
|
|
<PageHeader :title="t`Import Wizard`">
|
2023-02-14 23:48:20 +05:30
|
|
|
<DropdownWithActions
|
|
|
|
:actions="actions"
|
|
|
|
v-if="hasImporter && !complete"
|
2023-02-15 14:02:48 +05:30
|
|
|
:disabled="isMakingEntries"
|
2023-02-14 23:48:20 +05:30
|
|
|
:title="t`More`"
|
|
|
|
/>
|
2023-02-02 13:42:09 +05:30
|
|
|
<Button
|
|
|
|
v-if="hasImporter && !complete"
|
2023-02-14 23:48:20 +05:30
|
|
|
:title="t`Add Row`"
|
2023-02-15 14:02:48 +05:30
|
|
|
@click="() => importer.addRow()"
|
|
|
|
:disabled="isMakingEntries"
|
2023-02-14 23:48:20 +05:30
|
|
|
:icon="true"
|
|
|
|
>
|
|
|
|
<feather-icon name="plus" class="w-4 h-4" />
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
v-if="hasImporter && !complete"
|
|
|
|
:title="t`Save Template`"
|
2023-02-02 13:42:09 +05:30
|
|
|
@click="saveTemplate"
|
2023-02-14 23:48:20 +05:30
|
|
|
:icon="true"
|
2023-02-02 13:42:09 +05:30
|
|
|
>
|
2023-02-14 23:48:20 +05:30
|
|
|
<feather-icon name="download" class="w-4 h-4" />
|
2023-02-04 12:56:37 +05:30
|
|
|
</Button>
|
2023-02-02 13:42:09 +05:30
|
|
|
<Button
|
2023-02-04 12:56:37 +05:30
|
|
|
v-if="canImportData"
|
2023-02-14 23:48:20 +05:30
|
|
|
:title="t`Import Data`"
|
2023-02-02 13:42:09 +05:30
|
|
|
type="primary"
|
2023-02-04 12:56:37 +05:30
|
|
|
@click="importData"
|
2023-02-15 14:02:48 +05:30
|
|
|
:disabled="errorMessage.length > 0 || isMakingEntries"
|
2023-02-02 13:42:09 +05:30
|
|
|
>
|
2023-02-04 12:56:37 +05:30
|
|
|
{{ t`Import Data` }}
|
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
v-if="importType && !canImportData"
|
2023-02-14 23:48:20 +05:30
|
|
|
:title="t`Select File`"
|
2023-02-04 12:56:37 +05:30
|
|
|
type="primary"
|
|
|
|
@click="selectFile"
|
|
|
|
>
|
|
|
|
{{ t`Select File` }}
|
|
|
|
</Button>
|
2023-02-02 13:42:09 +05:30
|
|
|
</PageHeader>
|
|
|
|
|
|
|
|
<!-- Main Body of the Wizard -->
|
|
|
|
<div class="flex text-base w-full flex-col" v-if="!complete">
|
|
|
|
<!-- Select Import Type -->
|
|
|
|
<div
|
|
|
|
class="
|
2023-02-14 23:48:20 +05:30
|
|
|
h-row-largest
|
2023-02-02 13:42:09 +05:30
|
|
|
flex flex-row
|
|
|
|
justify-start
|
|
|
|
items-center
|
|
|
|
w-full
|
|
|
|
gap-2
|
|
|
|
border-b
|
|
|
|
p-4
|
|
|
|
"
|
|
|
|
>
|
2023-02-02 17:52:50 +05:30
|
|
|
<AutoComplete
|
|
|
|
:df="{
|
|
|
|
fieldname: 'importType',
|
|
|
|
label: t`Import Type`,
|
|
|
|
fieldtype: 'AutoComplete',
|
|
|
|
options: importableSchemaNames.map((value) => ({
|
|
|
|
value,
|
|
|
|
label: fyo.schemaMap[value]?.label ?? value,
|
|
|
|
})),
|
|
|
|
}"
|
2023-02-02 13:42:09 +05:30
|
|
|
input-class="bg-transparent text-gray-900 text-base"
|
2023-02-02 17:52:50 +05:30
|
|
|
class="w-40"
|
|
|
|
:border="true"
|
2023-02-02 13:42:09 +05:30
|
|
|
:value="importType"
|
|
|
|
size="small"
|
|
|
|
@change="setImportType"
|
|
|
|
/>
|
|
|
|
|
2023-02-14 14:56:09 +05:30
|
|
|
<p v-if="errorMessage.length > 0" class="text-base ms-2 text-red-500">
|
|
|
|
{{ errorMessage }}
|
|
|
|
</p>
|
2023-02-02 13:42:09 +05:30
|
|
|
<p
|
2023-02-14 14:56:09 +05:30
|
|
|
v-else
|
2023-02-02 13:42:09 +05:30
|
|
|
class="text-base ms-2"
|
|
|
|
:class="fileName ? 'text-gray-900 font-semibold' : 'text-gray-700'"
|
|
|
|
>
|
2023-02-15 13:29:06 +05:30
|
|
|
<span v-if="fileName" class="font-normal">{{ t`Selected` }} </span>
|
2023-02-14 14:56:09 +05:30
|
|
|
{{ helperMessage }}{{ fileName ? ',' : '' }}
|
2023-02-02 13:42:09 +05:30
|
|
|
<span v-if="fileName" class="font-normal">
|
2023-02-15 13:29:06 +05:30
|
|
|
{{ t`check values and click on` }} </span
|
2023-02-15 10:16:00 +05:30
|
|
|
>{{ ' ' }}<span v-if="fileName">{{ t`Import Data.` }}</span>
|
|
|
|
<span
|
|
|
|
v-if="hasImporter && importer.valueMatrix.length > 0"
|
|
|
|
class="font-normal"
|
2023-02-15 13:29:06 +05:30
|
|
|
>{{
|
|
|
|
' ' +
|
|
|
|
(importer.valueMatrix.length === 2
|
|
|
|
? t`${importer.valueMatrix.length} row added.`
|
|
|
|
: t`${importer.valueMatrix.length} rows added.`)
|
|
|
|
}}</span
|
2023-02-15 10:16:00 +05:30
|
|
|
>
|
2023-02-02 13:42:09 +05:30
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
2023-02-15 13:29:06 +05:30
|
|
|
<!-- Assignment Row and Value Grid container -->
|
|
|
|
<div
|
|
|
|
v-if="hasImporter"
|
|
|
|
class="overflow-auto custom-scroll"
|
|
|
|
style="max-height: calc(100vh - (2 * var(--h-row-largest)) - 2px)"
|
|
|
|
>
|
|
|
|
<!-- Column Assignment Row -->
|
2023-02-14 23:48:20 +05:30
|
|
|
<div
|
2023-02-15 13:29:06 +05:30
|
|
|
class="grid sticky top-0 py-4 pe-4 bg-white border-b gap-4"
|
|
|
|
style="z-index: 1; width: fit-content"
|
|
|
|
:style="gridTemplateColumn"
|
2023-02-14 23:48:20 +05:30
|
|
|
>
|
2023-02-15 13:29:06 +05:30
|
|
|
<div class="index-cell">#</div>
|
2023-02-15 14:02:48 +05:30
|
|
|
<Select
|
2023-02-15 13:29:06 +05:30
|
|
|
v-for="index in columnIterator"
|
|
|
|
class="flex-shrink-0"
|
|
|
|
size="small"
|
|
|
|
:border="true"
|
|
|
|
:key="index"
|
|
|
|
:df="gridColumnTitleDf"
|
|
|
|
:value="importer.assignedTemplateFields[index]"
|
|
|
|
@change="(value: string | null) => importer.setTemplateField(index, value)"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- Values Grid -->
|
|
|
|
<div
|
|
|
|
v-if="importer.valueMatrix.length"
|
|
|
|
class="grid py-4 pe-4 bg-white gap-4"
|
|
|
|
style="width: fit-content"
|
|
|
|
:style="gridTemplateColumn"
|
|
|
|
>
|
|
|
|
<!-- Grid Value Row Cells, Allow Editing Values -->
|
|
|
|
<template v-for="(row, ridx) of importer.valueMatrix" :key="ridx">
|
2023-02-02 13:42:09 +05:30
|
|
|
<div
|
2023-02-15 13:29:06 +05:30
|
|
|
class="index-cell group cursor-pointer"
|
|
|
|
@click="importer.removeRow(ridx)"
|
2023-02-02 13:42:09 +05:30
|
|
|
>
|
2023-02-15 13:29:06 +05:30
|
|
|
<feather-icon
|
|
|
|
name="x"
|
|
|
|
class="w-4 h-4 hidden group-hover:inline-block -me-1"
|
|
|
|
:button="true"
|
|
|
|
/>
|
2023-02-02 13:42:09 +05:30
|
|
|
<span class="group-hover:hidden">
|
2023-02-15 13:29:06 +05:30
|
|
|
{{ ridx + 1 }}
|
2023-02-02 13:42:09 +05:30
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
|
2023-02-15 13:29:06 +05:30
|
|
|
<template
|
|
|
|
v-for="(val, cidx) of row.slice(0, columnCount)"
|
|
|
|
:key="`cell-${ridx}-${cidx}`"
|
|
|
|
>
|
|
|
|
<!-- Raw Data Field if Column is Not Assigned -->
|
|
|
|
<Data
|
|
|
|
v-if="!importer.assignedTemplateFields[cidx]"
|
|
|
|
:title="getFieldTitle(val)"
|
|
|
|
:df="{
|
|
|
|
fieldname: 'tempField',
|
|
|
|
label: t`Temporary`,
|
|
|
|
placeholder: t`Select column`,
|
|
|
|
}"
|
|
|
|
size="small"
|
|
|
|
:border="true"
|
|
|
|
:value="
|
|
|
|
val.value != null
|
|
|
|
? String(val.value)
|
|
|
|
: val.rawValue != null
|
|
|
|
? String(val.rawValue)
|
|
|
|
: ''
|
|
|
|
"
|
|
|
|
:read-only="true"
|
|
|
|
/>
|
|
|
|
|
|
|
|
<!-- FormControl Field if Column is Assigned -->
|
|
|
|
<FormControl
|
|
|
|
v-else
|
|
|
|
:class="val.error ? 'border border-red-300 rounded-md' : ''"
|
|
|
|
:title="getFieldTitle(val)"
|
|
|
|
:df="
|
2023-02-02 13:42:09 +05:30
|
|
|
importer.templateFieldsMap.get(
|
2023-02-02 17:52:50 +05:30
|
|
|
importer.assignedTemplateFields[cidx]!
|
2023-02-02 13:42:09 +05:30
|
|
|
)
|
|
|
|
"
|
2023-02-15 13:29:06 +05:30
|
|
|
size="small"
|
|
|
|
:rows="1"
|
|
|
|
:border="true"
|
|
|
|
:value="val.error ? null : val.value"
|
|
|
|
@change="(value: DocValue)=> {
|
2023-02-02 17:52:50 +05:30
|
|
|
importer.valueMatrix[ridx][cidx]!.error = false
|
2023-02-02 13:42:09 +05:30
|
|
|
importer.valueMatrix[ridx][cidx]!.value = value
|
|
|
|
}"
|
2023-02-15 13:29:06 +05:30
|
|
|
/>
|
2023-02-02 13:42:09 +05:30
|
|
|
</template>
|
2023-02-15 13:29:06 +05:30
|
|
|
</template>
|
2023-02-02 13:42:09 +05:30
|
|
|
</div>
|
|
|
|
|
2023-02-15 14:02:48 +05:30
|
|
|
<div
|
|
|
|
v-else
|
|
|
|
class="ps-4 text-gray-700 sticky left-0 flex items-center"
|
|
|
|
style="height: 62.5px"
|
|
|
|
>
|
2023-02-15 13:29:06 +05:30
|
|
|
{{ t`No rows added. Select a file or add rows.` }}
|
|
|
|
</div>
|
|
|
|
</div>
|
2023-02-02 13:42:09 +05:30
|
|
|
</div>
|
|
|
|
|
2023-02-02 17:52:50 +05:30
|
|
|
<!-- Loading Bar when Saving Docs -->
|
2023-02-02 13:42:09 +05:30
|
|
|
<Loading
|
|
|
|
v-if="isMakingEntries"
|
|
|
|
:open="isMakingEntries"
|
|
|
|
:percent="percentLoading"
|
|
|
|
:message="messageLoading"
|
|
|
|
/>
|
2023-02-02 17:52:50 +05:30
|
|
|
|
|
|
|
<!-- Pick Column Modal -->
|
|
|
|
<Modal
|
|
|
|
:open-modal="showColumnPicker"
|
|
|
|
@closemodal="showColumnPicker = false"
|
|
|
|
>
|
|
|
|
<div class="w-form">
|
|
|
|
<!-- Pick Column Header -->
|
|
|
|
<FormHeader :form-title="t`Pick Import Columns`" />
|
|
|
|
<hr />
|
|
|
|
|
|
|
|
<!-- Pick Column Checkboxes -->
|
|
|
|
<div
|
|
|
|
class="p-4 max-h-80 overflow-auto custom-scroll"
|
|
|
|
v-for="[key, value] of columnPickerFieldsMap.entries()"
|
|
|
|
:key="key"
|
|
|
|
>
|
|
|
|
<h2 class="text-sm font-semibold text-gray-800">
|
|
|
|
{{ key }}
|
|
|
|
</h2>
|
|
|
|
<div class="grid grid-cols-3 border rounded mt-1">
|
|
|
|
<div
|
|
|
|
v-for="tf of value"
|
|
|
|
:key="tf.fieldKey"
|
|
|
|
class="flex items-center"
|
|
|
|
>
|
|
|
|
<Check
|
|
|
|
:df="{
|
|
|
|
fieldname: tf.fieldname,
|
|
|
|
label: tf.label,
|
|
|
|
}"
|
|
|
|
:show-label="true"
|
|
|
|
:read-only="tf.required"
|
|
|
|
:value="importer.templateFieldsPicked.get(tf.fieldKey)"
|
|
|
|
@change="(value:boolean) => pickColumn(tf.fieldKey, value)"
|
|
|
|
/>
|
|
|
|
<p v-if="tf.required" class="w-0 text-red-600 -ml-4">*</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- Pick Column Footer -->
|
|
|
|
<hr />
|
|
|
|
<div class="p-4 flex justify-between items-center">
|
|
|
|
<p class="text-sm text-gray-600">
|
|
|
|
{{ t`${numColumnsPicked} fields selected` }}
|
|
|
|
</p>
|
|
|
|
<Button type="primary" @click="showColumnPicker = false">{{
|
|
|
|
t`Done`
|
|
|
|
}}</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Modal>
|
2023-02-15 10:16:00 +05:30
|
|
|
|
|
|
|
<!-- Import Completed Modal -->
|
|
|
|
<Modal :open-modal="complete" @closemodal="clear">
|
|
|
|
<div class="w-form">
|
|
|
|
<!-- Import Completed Header -->
|
|
|
|
<FormHeader :form-title="t`Import Complete`" />
|
|
|
|
<hr />
|
|
|
|
<!-- Success -->
|
|
|
|
<div v-if="success.length > 0">
|
|
|
|
<!-- Success Section Header -->
|
|
|
|
<div class="flex justify-between px-4 pt-4 pb-1">
|
|
|
|
<p class="text-base font-semibold">{{ t`Success` }}</p>
|
|
|
|
<p class="text-sm text-gray-600">
|
|
|
|
{{
|
|
|
|
success.length === 1
|
|
|
|
? t`${success.length} entry imported`
|
|
|
|
: t`${success.length} entries imported`
|
|
|
|
}}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
<!-- Success Body -->
|
|
|
|
<div class="max-h-40 overflow-auto text-gray-900">
|
|
|
|
<div
|
|
|
|
v-for="(name, i) of success"
|
|
|
|
:key="name"
|
|
|
|
class="px-4 py-1 grid grid-cols-2 text-base gap-4"
|
|
|
|
style="grid-template-columns: 1rem auto"
|
|
|
|
>
|
|
|
|
<div class="text-end">{{ i + 1 }}.</div>
|
|
|
|
<p class="whitespace-nowrap overflow-auto no-scrollbar">
|
|
|
|
{{ name }}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<hr />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- Failed -->
|
|
|
|
<div v-if="failed.length > 0">
|
|
|
|
<!-- Failed Section Header -->
|
|
|
|
<div class="flex justify-between px-4 pt-4 pb-1">
|
|
|
|
<p class="text-base font-semibold">{{ t`Failed` }}</p>
|
|
|
|
<p class="text-sm text-gray-600">
|
|
|
|
{{
|
|
|
|
failed.length === 1
|
|
|
|
? t`${failed.length} entry failed`
|
|
|
|
: t`${failed.length} entries failed`
|
|
|
|
}}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
<!-- Failed Body -->
|
|
|
|
<div class="max-h-40 overflow-auto text-gray-900">
|
|
|
|
<div
|
|
|
|
v-for="(f, i) of failed"
|
|
|
|
:key="f.name"
|
|
|
|
class="px-4 py-1 grid grid-cols-2 text-base gap-4"
|
|
|
|
style="grid-template-columns: 1rem 8rem auto"
|
|
|
|
>
|
|
|
|
<div class="text-end">{{ i + 1 }}.</div>
|
|
|
|
<p class="whitespace-nowrap overflow-auto no-scrollbar">
|
|
|
|
{{ f.name }}
|
|
|
|
</p>
|
|
|
|
<p class="whitespace-nowrap overflow-auto no-scrollbar">
|
|
|
|
{{ f.error.message }}
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<hr />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- Fallback Div -->
|
|
|
|
<div
|
|
|
|
v-if="failed.length === 0 && success.length === 0"
|
|
|
|
class="p-4 text-base"
|
|
|
|
>
|
|
|
|
{{ t`No entries were imported.` }}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- Footer Button -->
|
|
|
|
<div class="flex justify-between p-4">
|
|
|
|
<Button
|
|
|
|
v-if="failed.length > 0"
|
|
|
|
@click="clearSuccessfullyImportedEntries"
|
|
|
|
>{{ t`Fix Failed` }}</Button
|
|
|
|
>
|
|
|
|
<Button
|
|
|
|
v-if="failed.length === 0 && success.length > 0"
|
|
|
|
@click="showMe"
|
|
|
|
>{{ t`Show Me` }}</Button
|
|
|
|
>
|
|
|
|
<Button @click="clear">{{ t`Done` }}</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Modal>
|
2023-02-02 13:42:09 +05:30
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
|
|
import { DocValue } from 'fyo/core/types';
|
2023-02-13 12:29:47 +05:30
|
|
|
import { RawValue } from 'schemas/types';
|
2023-02-02 13:42:09 +05:30
|
|
|
import { Action as BaseAction } from 'fyo/model/types';
|
|
|
|
import { ValidationError } from 'fyo/utils/errors';
|
|
|
|
import { ModelNameEnum } from 'models/types';
|
2023-02-02 17:52:50 +05:30
|
|
|
import { OptionField, SelectOption } from 'schemas/types';
|
2023-02-02 13:42:09 +05:30
|
|
|
import Button from 'src/components/Button.vue';
|
|
|
|
import AutoComplete from 'src/components/Controls/AutoComplete.vue';
|
2023-02-02 17:52:50 +05:30
|
|
|
import Check from 'src/components/Controls/Check.vue';
|
|
|
|
import Data from 'src/components/Controls/Data.vue';
|
2023-02-02 13:42:09 +05:30
|
|
|
import FormControl from 'src/components/Controls/FormControl.vue';
|
|
|
|
import DropdownWithActions from 'src/components/DropdownWithActions.vue';
|
2023-02-02 17:52:50 +05:30
|
|
|
import FormHeader from 'src/components/FormHeader.vue';
|
|
|
|
import Modal from 'src/components/Modal.vue';
|
2023-02-02 13:42:09 +05:30
|
|
|
import PageHeader from 'src/components/PageHeader.vue';
|
2023-02-14 14:56:09 +05:30
|
|
|
import { getColumnLabel, Importer, TemplateField } from 'src/importer';
|
2023-02-02 13:42:09 +05:30
|
|
|
import { fyo } from 'src/initFyo';
|
|
|
|
import { getSavePath, saveData, selectFile } from 'src/utils/ipcCalls';
|
|
|
|
import { docsPathMap } from 'src/utils/misc';
|
|
|
|
import { docsPathRef } from 'src/utils/refs';
|
|
|
|
import { showMessageDialog } from 'src/utils/ui';
|
|
|
|
import { defineComponent } from 'vue';
|
|
|
|
import Loading from '../components/Loading.vue';
|
2023-02-15 14:02:48 +05:30
|
|
|
import Select from 'src/components/Controls/Select.vue';
|
|
|
|
import { isWeakMap } from 'lodash';
|
2023-02-02 13:42:09 +05:30
|
|
|
|
|
|
|
type Action = Pick<BaseAction, 'condition' | 'component'> & {
|
|
|
|
action: Function;
|
|
|
|
};
|
|
|
|
|
2023-02-14 15:05:09 +05:30
|
|
|
type ImportWizardData = {
|
2023-02-02 17:52:50 +05:30
|
|
|
showColumnPicker: boolean;
|
2023-02-02 13:42:09 +05:30
|
|
|
complete: boolean;
|
2023-02-15 10:16:00 +05:30
|
|
|
success: string[];
|
2023-02-15 10:41:22 +05:30
|
|
|
successOldName: string[];
|
2023-02-13 20:48:56 +05:30
|
|
|
failed: { name: string; error: Error }[];
|
2023-02-02 13:42:09 +05:30
|
|
|
file: null | { name: string; filePath: string; text: string };
|
|
|
|
nullOrImporter: null | Importer;
|
|
|
|
importType: string;
|
|
|
|
isMakingEntries: boolean;
|
|
|
|
percentLoading: number;
|
|
|
|
messageLoading: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export default defineComponent({
|
|
|
|
components: {
|
|
|
|
PageHeader,
|
|
|
|
FormControl,
|
|
|
|
Button,
|
|
|
|
DropdownWithActions,
|
|
|
|
Loading,
|
|
|
|
AutoComplete,
|
2023-02-02 17:52:50 +05:30
|
|
|
Data,
|
|
|
|
Modal,
|
|
|
|
FormHeader,
|
|
|
|
Check,
|
2023-02-15 14:02:48 +05:30
|
|
|
Select,
|
2023-02-02 13:42:09 +05:30
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
2023-02-02 17:52:50 +05:30
|
|
|
showColumnPicker: false,
|
2023-02-02 13:42:09 +05:30
|
|
|
complete: false,
|
2023-02-15 10:16:00 +05:30
|
|
|
success: [],
|
2023-02-15 10:41:22 +05:30
|
|
|
successOldName: [],
|
2023-02-13 20:48:56 +05:30
|
|
|
failed: [],
|
2023-02-02 13:42:09 +05:30
|
|
|
file: null,
|
|
|
|
nullOrImporter: null,
|
|
|
|
importType: '',
|
|
|
|
isMakingEntries: false,
|
|
|
|
percentLoading: 0,
|
|
|
|
messageLoading: '',
|
2023-02-14 15:05:09 +05:30
|
|
|
} as ImportWizardData;
|
2023-02-02 13:42:09 +05:30
|
|
|
},
|
|
|
|
mounted() {
|
|
|
|
if (fyo.store.isDevelopment) {
|
|
|
|
// @ts-ignore
|
2023-02-02 17:52:50 +05:30
|
|
|
window.iw = this;
|
|
|
|
this.setImportType('Item');
|
2023-02-02 13:42:09 +05:30
|
|
|
}
|
|
|
|
},
|
2023-02-15 14:02:48 +05:30
|
|
|
watch: {
|
|
|
|
columnCount(val) {
|
|
|
|
if (!this.hasImporter) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const possiblyAssigned = this.importer.assignedTemplateFields.length;
|
|
|
|
if (val >= this.importer.assignedTemplateFields.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = val; i < possiblyAssigned; i++) {
|
|
|
|
this.importer.assignedTemplateFields[i] = null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
computed: {
|
2023-02-15 13:29:06 +05:30
|
|
|
gridTemplateColumn(): string {
|
|
|
|
return `grid-template-columns: 4rem repeat(${this.columnCount}, 10rem)`;
|
|
|
|
},
|
2023-02-14 14:56:09 +05:30
|
|
|
duplicates(): string[] {
|
|
|
|
if (!this.hasImporter) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
const dupes = new Set<string>();
|
|
|
|
const assignedSet = new Set<string>();
|
|
|
|
|
|
|
|
for (const key of this.importer.assignedTemplateFields) {
|
|
|
|
if (!key) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const tf = this.importer.templateFieldsMap.get(key);
|
|
|
|
if (assignedSet.has(key) && tf) {
|
|
|
|
dupes.add(getColumnLabel(tf));
|
|
|
|
}
|
|
|
|
|
|
|
|
assignedSet.add(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Array.from(dupes);
|
|
|
|
},
|
|
|
|
requiredNotSelected(): string[] {
|
|
|
|
if (!this.hasImporter) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
const assigned = new Set(this.importer.assignedTemplateFields);
|
|
|
|
return [...this.importer.templateFieldsMap.values()]
|
|
|
|
.filter((f) => f.required && !assigned.has(f.fieldKey))
|
|
|
|
.map((f) => getColumnLabel(f));
|
|
|
|
},
|
|
|
|
errorMessage(): string {
|
|
|
|
if (this.duplicates.length) {
|
|
|
|
return this.t`Duplicate columns found: ${this.duplicates.join(', ')}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.requiredNotSelected.length) {
|
|
|
|
return this
|
|
|
|
.t`Required fields not selected: ${this.requiredNotSelected.join(
|
|
|
|
', '
|
|
|
|
)}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return '';
|
|
|
|
},
|
2023-02-04 12:56:37 +05:30
|
|
|
canImportData(): boolean {
|
2023-02-13 20:48:56 +05:30
|
|
|
if (!this.hasImporter) {
|
|
|
|
return false;
|
2023-02-04 12:56:37 +05:30
|
|
|
}
|
|
|
|
|
2023-02-13 20:48:56 +05:30
|
|
|
return this.importer.valueMatrix.length > 0;
|
2023-02-04 12:56:37 +05:30
|
|
|
},
|
|
|
|
canSelectFile(): boolean {
|
|
|
|
return !this.file;
|
|
|
|
},
|
2023-02-02 17:52:50 +05:30
|
|
|
columnCount(): number {
|
2023-02-15 14:02:48 +05:30
|
|
|
if (!this.hasImporter) {
|
|
|
|
return 0;
|
2023-02-02 17:52:50 +05:30
|
|
|
}
|
|
|
|
|
2023-02-04 12:56:37 +05:30
|
|
|
if (!this.file) {
|
|
|
|
return this.numColumnsPicked;
|
|
|
|
}
|
|
|
|
|
2023-02-15 14:02:48 +05:30
|
|
|
let vmColumnCount = 0;
|
|
|
|
if (this.importer.valueMatrix.length) {
|
|
|
|
vmColumnCount = this.importer.valueMatrix[0].length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Math.min(
|
|
|
|
this.importer.assignedTemplateFields.length,
|
|
|
|
vmColumnCount
|
|
|
|
);
|
2023-02-02 17:52:50 +05:30
|
|
|
},
|
|
|
|
columnIterator(): number[] {
|
|
|
|
return Array(this.columnCount)
|
|
|
|
.fill(null)
|
|
|
|
.map((_, i) => i);
|
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
hasImporter(): boolean {
|
|
|
|
return !!this.nullOrImporter;
|
|
|
|
},
|
2023-02-02 17:52:50 +05:30
|
|
|
numColumnsPicked(): number {
|
|
|
|
return [...this.importer.templateFieldsPicked.values()].filter(Boolean)
|
|
|
|
.length;
|
|
|
|
},
|
|
|
|
columnPickerFieldsMap(): Map<string, TemplateField[]> {
|
|
|
|
const map: Map<string, TemplateField[]> = new Map();
|
|
|
|
|
|
|
|
for (const value of this.importer.templateFieldsMap.values()) {
|
|
|
|
let label = value.schemaLabel;
|
|
|
|
if (value.parentSchemaChildField) {
|
|
|
|
label = `${value.parentSchemaChildField.label} (${value.schemaLabel})`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!map.has(label)) {
|
|
|
|
map.set(label, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
map.get(label)!.push(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return map;
|
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
importer(): Importer {
|
|
|
|
if (!this.nullOrImporter) {
|
|
|
|
throw new ValidationError(this.t`Importer not set, reload tool`, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.nullOrImporter as Importer;
|
|
|
|
},
|
|
|
|
importableSchemaNames(): ModelNameEnum[] {
|
|
|
|
const importables = [
|
|
|
|
ModelNameEnum.SalesInvoice,
|
|
|
|
ModelNameEnum.PurchaseInvoice,
|
|
|
|
ModelNameEnum.Payment,
|
|
|
|
ModelNameEnum.Party,
|
|
|
|
ModelNameEnum.Item,
|
|
|
|
ModelNameEnum.JournalEntry,
|
|
|
|
];
|
|
|
|
|
|
|
|
const hasInventory = fyo.doc.singles.AccountingSettings?.enableInventory;
|
|
|
|
if (hasInventory) {
|
|
|
|
importables.push(
|
|
|
|
ModelNameEnum.StockMovement,
|
|
|
|
ModelNameEnum.Shipment,
|
|
|
|
ModelNameEnum.PurchaseReceipt,
|
|
|
|
ModelNameEnum.Location
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return importables;
|
|
|
|
},
|
|
|
|
actions(): Action[] {
|
|
|
|
const actions: Action[] = [];
|
|
|
|
|
2023-02-04 12:56:37 +05:30
|
|
|
let selectFileLabel = this.t`Select File`;
|
2023-02-02 13:42:09 +05:30
|
|
|
if (this.file) {
|
2023-02-04 12:56:37 +05:30
|
|
|
selectFileLabel = this.t`Change File`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.canImportData) {
|
2023-02-02 13:42:09 +05:30
|
|
|
actions.push({
|
|
|
|
component: {
|
2023-02-04 12:56:37 +05:30
|
|
|
template: `<span>{{ "${selectFileLabel}" }}</span>`,
|
2023-02-02 13:42:09 +05:30
|
|
|
},
|
|
|
|
action: this.selectFile,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-02-02 17:52:50 +05:30
|
|
|
const pickColumnsAction = {
|
|
|
|
component: {
|
|
|
|
template: '<span>{{ t`Pick Import Columns` }}</span>',
|
|
|
|
},
|
|
|
|
action: () => (this.showColumnPicker = true),
|
|
|
|
};
|
|
|
|
|
2023-02-02 13:42:09 +05:30
|
|
|
const cancelAction = {
|
|
|
|
component: {
|
|
|
|
template: '<span class="text-red-700" >{{ t`Cancel` }}</span>',
|
|
|
|
},
|
|
|
|
action: this.clear,
|
|
|
|
};
|
2023-02-02 17:52:50 +05:30
|
|
|
actions.push(pickColumnsAction, cancelAction);
|
2023-02-02 13:42:09 +05:30
|
|
|
|
|
|
|
return actions;
|
|
|
|
},
|
|
|
|
fileName(): string {
|
|
|
|
if (!this.file) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.file.name;
|
|
|
|
},
|
2023-02-14 14:56:09 +05:30
|
|
|
helperMessage(): string {
|
2023-02-02 13:42:09 +05:30
|
|
|
if (!this.importType) {
|
|
|
|
return this.t`Set an Import Type`;
|
|
|
|
} else if (!this.fileName) {
|
2023-02-15 13:29:06 +05:30
|
|
|
return '';
|
2023-02-02 13:42:09 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return this.fileName;
|
|
|
|
},
|
|
|
|
isSubmittable(): boolean {
|
|
|
|
const schemaName = this.importer.schemaName;
|
|
|
|
return fyo.schemaMap[schemaName]?.isSubmittable ?? false;
|
|
|
|
},
|
|
|
|
gridColumnTitleDf(): OptionField {
|
|
|
|
const options: SelectOption[] = [];
|
|
|
|
for (const field of this.importer.templateFieldsMap.values()) {
|
|
|
|
const value = field.fieldKey;
|
2023-02-02 17:52:50 +05:30
|
|
|
if (!this.importer.templateFieldsPicked.get(value)) {
|
|
|
|
continue;
|
|
|
|
}
|
2023-02-02 13:42:09 +05:30
|
|
|
|
2023-02-14 14:56:09 +05:30
|
|
|
const label = getColumnLabel(field);
|
2023-02-02 13:42:09 +05:30
|
|
|
|
|
|
|
options.push({ value, label });
|
|
|
|
}
|
|
|
|
|
2023-02-15 14:02:48 +05:30
|
|
|
options.push({ value: '', label: this.t`None` });
|
2023-02-02 13:42:09 +05:30
|
|
|
return {
|
|
|
|
fieldname: 'col',
|
2023-02-15 14:02:48 +05:30
|
|
|
fieldtype: 'Select',
|
2023-02-02 13:42:09 +05:30
|
|
|
options,
|
|
|
|
} as OptionField;
|
|
|
|
},
|
|
|
|
pickedArray(): string[] {
|
|
|
|
return [...this.importer.templateFieldsPicked.entries()]
|
2023-02-02 17:52:50 +05:30
|
|
|
.filter(([_, picked]) => picked)
|
2023-02-02 13:42:09 +05:30
|
|
|
.map(([key, _]) => key);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
activated(): void {
|
2023-02-14 15:05:09 +05:30
|
|
|
docsPathRef.value = docsPathMap.ImportWizard ?? '';
|
2023-02-02 13:42:09 +05:30
|
|
|
},
|
|
|
|
deactivated(): void {
|
|
|
|
docsPathRef.value = '';
|
|
|
|
if (!this.complete) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.clear();
|
|
|
|
},
|
|
|
|
methods: {
|
2023-02-13 12:29:47 +05:30
|
|
|
getFieldTitle(vmi: {
|
|
|
|
value?: DocValue;
|
|
|
|
rawValue?: RawValue;
|
|
|
|
error?: boolean;
|
2023-02-14 14:56:09 +05:30
|
|
|
}): string {
|
2023-02-13 12:29:47 +05:30
|
|
|
const title: string[] = [];
|
|
|
|
if (vmi.value != null) {
|
|
|
|
title.push(this.t`Value: ${String(vmi.value)}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vmi.rawValue != null) {
|
|
|
|
title.push(this.t`Raw Value: ${String(vmi.rawValue)}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vmi.error) {
|
|
|
|
title.push(this.t`Conversion Error`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!title.length) {
|
2023-02-14 14:56:09 +05:30
|
|
|
return this.t`No Value`;
|
2023-02-13 12:29:47 +05:30
|
|
|
}
|
2023-02-14 14:56:09 +05:30
|
|
|
|
2023-02-13 12:29:47 +05:30
|
|
|
return title.join(', ');
|
|
|
|
},
|
2023-02-02 17:52:50 +05:30
|
|
|
pickColumn(fieldKey: string, value: boolean): void {
|
|
|
|
this.importer.templateFieldsPicked.set(fieldKey, value);
|
|
|
|
if (value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const idx = this.importer.assignedTemplateFields.findIndex(
|
|
|
|
(f) => f === fieldKey
|
|
|
|
);
|
|
|
|
|
|
|
|
if (idx >= 0) {
|
|
|
|
this.importer.assignedTemplateFields[idx] = null;
|
2023-02-04 12:56:37 +05:30
|
|
|
this.reassignTemplateFields();
|
|
|
|
}
|
|
|
|
},
|
2023-02-14 14:56:09 +05:30
|
|
|
reassignTemplateFields(): void {
|
2023-02-04 12:56:37 +05:30
|
|
|
if (this.importer.valueMatrix.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const idx in this.importer.assignedTemplateFields) {
|
|
|
|
this.importer.assignedTemplateFields[idx] = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
let idx = 0;
|
|
|
|
for (const [fieldKey, value] of this.importer.templateFieldsPicked) {
|
|
|
|
if (!value) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.importer.assignedTemplateFields[idx] = fieldKey;
|
|
|
|
idx += 1;
|
2023-02-02 17:52:50 +05:30
|
|
|
}
|
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
showMe(): void {
|
|
|
|
const schemaName = this.importer.schemaName;
|
|
|
|
this.clear();
|
|
|
|
this.$router.push(`/list/${schemaName}`);
|
|
|
|
},
|
|
|
|
clear(): void {
|
|
|
|
this.file = null;
|
2023-02-15 10:16:00 +05:30
|
|
|
this.success = [];
|
2023-02-15 10:41:22 +05:30
|
|
|
this.successOldName = [];
|
2023-02-13 20:48:56 +05:30
|
|
|
this.failed = [];
|
2023-02-02 13:42:09 +05:30
|
|
|
this.nullOrImporter = null;
|
|
|
|
this.importType = '';
|
|
|
|
this.complete = false;
|
|
|
|
this.isMakingEntries = false;
|
|
|
|
this.percentLoading = 0;
|
|
|
|
this.messageLoading = '';
|
|
|
|
},
|
|
|
|
async saveTemplate(): Promise<void> {
|
|
|
|
const template = this.importer.getCSVTemplate();
|
|
|
|
const templateName = this.importType + ' ' + this.t`Template`;
|
|
|
|
const { canceled, filePath } = await getSavePath(templateName, 'csv');
|
|
|
|
|
|
|
|
if (canceled || !filePath) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await saveData(template, filePath);
|
|
|
|
},
|
2023-02-14 14:56:09 +05:30
|
|
|
async preImportValidations(): Promise<boolean> {
|
|
|
|
const message = this.t`Cannot Import`;
|
|
|
|
if (this.errorMessage.length) {
|
|
|
|
await showMessageDialog({
|
|
|
|
message,
|
|
|
|
detail: this.errorMessage,
|
|
|
|
});
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const cellErrors = this.importer.checkCellErrors();
|
|
|
|
if (cellErrors.length) {
|
|
|
|
await showMessageDialog({
|
|
|
|
message,
|
|
|
|
detail: this.t`Following cells have errors: ${cellErrors.join(', ')}`,
|
|
|
|
});
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const absentLinks = await this.importer.checkLinks();
|
|
|
|
if (absentLinks.length) {
|
|
|
|
await showMessageDialog({
|
|
|
|
message,
|
|
|
|
detail: this.t`Following links do not exist: ${absentLinks
|
|
|
|
.map((l) => `(${l.schemaLabel}, ${l.name})`)
|
|
|
|
.join(', ')}`,
|
|
|
|
});
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
async importData(): Promise<void> {
|
2023-02-14 14:56:09 +05:30
|
|
|
const isValid = await this.preImportValidations();
|
|
|
|
if (!isValid || this.isMakingEntries || this.complete) {
|
2023-02-02 13:42:09 +05:30
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-02-15 10:16:00 +05:30
|
|
|
this.isMakingEntries = true;
|
2023-02-14 13:34:22 +05:30
|
|
|
this.importer.populateDocs();
|
2023-02-02 13:42:09 +05:30
|
|
|
|
2023-02-15 10:41:22 +05:30
|
|
|
const shouldSubmit = await this.askShouldSubmit();
|
|
|
|
|
2023-02-13 20:48:56 +05:30
|
|
|
let doneCount = 0;
|
|
|
|
for (const doc of this.importer.docs) {
|
|
|
|
this.setLoadingStatus(doneCount, this.importer.docs.length);
|
2023-02-15 10:41:22 +05:30
|
|
|
const oldName = doc.name ?? '';
|
2023-02-13 20:48:56 +05:30
|
|
|
try {
|
|
|
|
await doc.sync();
|
2023-02-15 10:41:22 +05:30
|
|
|
if (shouldSubmit) {
|
|
|
|
await doc.submit();
|
|
|
|
}
|
2023-02-13 20:48:56 +05:30
|
|
|
doneCount += 1;
|
2023-02-02 13:42:09 +05:30
|
|
|
|
2023-02-15 10:16:00 +05:30
|
|
|
this.success.push(doc.name!);
|
2023-02-15 10:41:22 +05:30
|
|
|
this.successOldName.push(oldName);
|
2023-02-13 20:48:56 +05:30
|
|
|
} catch (error) {
|
|
|
|
if (error instanceof Error) {
|
|
|
|
this.failed.push({ name: doc.name!, error });
|
|
|
|
}
|
|
|
|
}
|
2023-02-02 13:42:09 +05:30
|
|
|
}
|
|
|
|
|
2023-02-13 20:48:56 +05:30
|
|
|
this.isMakingEntries = false;
|
2023-02-02 13:42:09 +05:30
|
|
|
this.complete = true;
|
|
|
|
},
|
2023-02-15 10:41:22 +05:30
|
|
|
async askShouldSubmit(): Promise<boolean> {
|
|
|
|
if (!this.fyo.schemaMap[this.importType]?.isSubmittable) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let shouldSubmit = false;
|
|
|
|
await showMessageDialog({
|
|
|
|
message: this.t`Should entries be submitted after syncing?`,
|
|
|
|
buttons: [
|
|
|
|
{
|
|
|
|
label: this.t`Yes`,
|
|
|
|
action() {
|
|
|
|
shouldSubmit = true;
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: this.t`No`,
|
|
|
|
action() {},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
return shouldSubmit;
|
|
|
|
},
|
2023-02-15 10:16:00 +05:30
|
|
|
clearSuccessfullyImportedEntries() {
|
2023-02-15 10:41:22 +05:30
|
|
|
const schemaName = this.importer.schemaName;
|
|
|
|
const nameFieldKey = `${schemaName}.name`;
|
|
|
|
const nameIndex = this.importer.assignedTemplateFields.findIndex(
|
|
|
|
(n) => n === nameFieldKey
|
|
|
|
);
|
|
|
|
|
|
|
|
const failedEntriesValueMatrix = this.importer.valueMatrix.filter(
|
|
|
|
(row) => {
|
|
|
|
const value = row[nameIndex].value;
|
|
|
|
if (typeof value !== 'string') {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return !this.successOldName.includes(value);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
this.setImportType(this.importType);
|
|
|
|
this.importer.valueMatrix = failedEntriesValueMatrix;
|
2023-02-15 10:16:00 +05:30
|
|
|
},
|
2023-02-02 13:42:09 +05:30
|
|
|
setImportType(importType: string): void {
|
|
|
|
this.clear();
|
|
|
|
if (!importType) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.importType = importType;
|
2023-02-02 17:52:50 +05:30
|
|
|
this.nullOrImporter = new Importer(importType, fyo);
|
2023-02-02 13:42:09 +05:30
|
|
|
},
|
2023-02-13 20:48:56 +05:30
|
|
|
setLoadingStatus(entriesMade: number, totalEntries: number): void {
|
2023-02-02 13:42:09 +05:30
|
|
|
this.percentLoading = entriesMade / totalEntries;
|
2023-02-13 20:48:56 +05:30
|
|
|
this.messageLoading = this.isMakingEntries
|
2023-02-02 13:42:09 +05:30
|
|
|
? `${entriesMade} entries made out of ${totalEntries}...`
|
|
|
|
: '';
|
|
|
|
},
|
|
|
|
async selectFile(): Promise<void> {
|
|
|
|
const options = {
|
|
|
|
title: this.t`Select File`,
|
|
|
|
filters: [{ name: 'CSV', extensions: ['csv'] }],
|
|
|
|
};
|
|
|
|
|
|
|
|
const { success, canceled, filePath, data, name } = await selectFile(
|
|
|
|
options
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!success && !canceled) {
|
|
|
|
await showMessageDialog({
|
|
|
|
message: this.t`File selection failed.`,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!success || canceled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const text = new TextDecoder().decode(data);
|
|
|
|
const isValid = this.importer.selectFile(text);
|
|
|
|
if (!isValid) {
|
|
|
|
await showMessageDialog({
|
|
|
|
message: this.t`Bad import data`,
|
|
|
|
detail: this.t`Could not read file`,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.file = {
|
|
|
|
name,
|
|
|
|
filePath,
|
|
|
|
text,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
</script>
|
2023-02-15 13:29:06 +05:30
|
|
|
<style scoped>
|
|
|
|
.index-cell {
|
|
|
|
@apply flex pe-4 justify-end items-center border-e bg-white sticky left-0 -my-4 text-gray-600;
|
|
|
|
}
|
|
|
|
</style>
|