2
0
mirror of https://github.com/frappe/books.git synced 2025-04-03 16:51:51 +00:00

fix: support option field value or label

- remove address placeholder
- allow more types of entries to be imported
This commit is contained in:
18alantom 2023-02-15 19:19:17 +05:30
parent aa04e1493c
commit 48797dff62
7 changed files with 74 additions and 20 deletions

View File

@ -73,7 +73,6 @@
"label": "Address", "label": "Address",
"fieldtype": "Link", "fieldtype": "Link",
"target": "Address", "target": "Address",
"placeholder": "Click to create",
"inline": true "inline": true
}, },
{ {

View File

@ -40,7 +40,6 @@
"label": "Address", "label": "Address",
"fieldtype": "Link", "fieldtype": "Link",
"target": "Address", "target": "Address",
"placeholder": "Click to create",
"inline": true "inline": true
}, },
{ {

View File

@ -16,7 +16,6 @@
"label": "Address", "label": "Address",
"fieldtype": "Link", "fieldtype": "Link",
"target": "Address", "target": "Address",
"placeholder": "Click to create",
"inline": true "inline": true
} }
], ],

View File

@ -73,7 +73,6 @@
"label": "Address", "label": "Address",
"fieldtype": "Link", "fieldtype": "Link",
"target": "Address", "target": "Address",
"placeholder": "Click to create",
"inline": true "inline": true
} }
], ],

View File

@ -8,13 +8,17 @@ import {
Field, Field,
FieldType, FieldType,
FieldTypeEnum, FieldTypeEnum,
OptionField,
RawValue, RawValue,
Schema, Schema,
TargetField, TargetField,
} from 'schemas/types'; } from 'schemas/types';
import { generateCSV, parseCSV } from 'utils/csvParser'; import { generateCSV, parseCSV } from 'utils/csvParser';
import { getValueMapFromList } from 'utils/index';
export type TemplateField = Field & { export type TemplateField = Field & TemplateFieldProps;
type TemplateFieldProps = {
schemaName: string; schemaName: string;
schemaLabel: string; schemaLabel: string;
fieldKey: string; fieldKey: string;
@ -82,6 +86,15 @@ export class Importer {
*/ */
docs: Doc[]; docs: Doc[];
/**
* Used if an options field is imported where the import data
* provided maybe the label and not the value
*/
optionsMap: {
values: Record<string, Set<string>>;
labelValueMap: Record<string, Record<string, string>>;
};
constructor(schemaName: string, fyo: Fyo) { constructor(schemaName: string, fyo: Fyo) {
if (!fyo.schemaMap[schemaName]) { if (!fyo.schemaMap[schemaName]) {
throw new ValidationError( throw new ValidationError(
@ -94,6 +107,10 @@ export class Importer {
this.fyo = fyo; this.fyo = fyo;
this.docs = []; this.docs = [];
this.valueMatrix = []; this.valueMatrix = [];
this.optionsMap = {
values: {},
labelValueMap: {},
};
const templateFields = getTemplateFields(schemaName, fyo, this); const templateFields = getTemplateFields(schemaName, fyo, this);
this.assignedTemplateFields = templateFields.map((f) => f.fieldKey); this.assignedTemplateFields = templateFields.map((f) => f.fieldKey);
@ -423,6 +440,10 @@ export class Importer {
return vmi; return vmi;
} }
if ('options' in tf && typeof vmi.rawValue === 'string') {
return this.getOptionFieldVmi(vmi, tf);
}
try { try {
vmi.value = Converter.toDocValue(rawValue, tf, this.fyo); vmi.value = Converter.toDocValue(rawValue, tf, this.fyo);
} catch { } catch {
@ -432,6 +453,39 @@ export class Importer {
return vmi; return vmi;
} }
getOptionFieldVmi(
{ rawValue }: ValueMatrixItem,
tf: OptionField & TemplateFieldProps
): ValueMatrixItem {
if (typeof rawValue !== 'string') {
return { error: true, value: null, rawValue };
}
if (!tf?.options.length) {
return { value: null, rawValue };
}
if (!this.optionsMap.labelValueMap[tf.fieldKey]) {
const values = new Set(tf.options.map(({ value }) => value));
const labelValueMap = getValueMapFromList(tf.options, 'label', 'value');
this.optionsMap.labelValueMap[tf.fieldKey] = labelValueMap;
this.optionsMap.values[tf.fieldKey] = values;
}
const hasValue = this.optionsMap.values[tf.fieldKey].has(rawValue);
if (hasValue) {
return { value: rawValue, rawValue };
}
const value = this.optionsMap.labelValueMap[tf.fieldKey][rawValue];
if (value) {
return { value, rawValue };
}
return { error: true, value: null, rawValue };
}
assignTemplateFieldsFromParsedRow(row: string[]): boolean { assignTemplateFieldsFromParsedRow(row: string[]): boolean {
const isKeyRow = row.some((key) => this.templateFieldsMap.has(key)); const isKeyRow = row.some((key) => this.templateFieldsMap.has(key));
if (!isKeyRow) { if (!isKeyRow) {
@ -550,6 +604,10 @@ function getTemplateFields(
tf.readOnly = false; tf.readOnly = false;
} }
if (schema.isChild && tf.fieldname === 'name') {
tf.required = false;
}
const schemaName = schema.name; const schemaName = schema.name;
const schemaLabel = schema.label; const schemaLabel = schema.label;
const fieldKey = `${schema.name}.${field.fieldname}`; const fieldKey = `${schema.name}.${field.fieldname}`;

View File

@ -4,12 +4,12 @@
<PageHeader :title="t`Import Wizard`"> <PageHeader :title="t`Import Wizard`">
<DropdownWithActions <DropdownWithActions
:actions="actions" :actions="actions"
v-if="hasImporter && !complete" v-if="hasImporter"
:disabled="isMakingEntries" :disabled="isMakingEntries"
:title="t`More`" :title="t`More`"
/> />
<Button <Button
v-if="hasImporter && !complete" v-if="hasImporter"
:title="t`Add Row`" :title="t`Add Row`"
@click="() => importer.addRow()" @click="() => importer.addRow()"
:disabled="isMakingEntries" :disabled="isMakingEntries"
@ -18,7 +18,7 @@
<feather-icon name="plus" class="w-4 h-4" /> <feather-icon name="plus" class="w-4 h-4" />
</Button> </Button>
<Button <Button
v-if="hasImporter && !complete" v-if="hasImporter"
:title="t`Save Template`" :title="t`Save Template`"
@click="saveTemplate" @click="saveTemplate"
:icon="true" :icon="true"
@ -45,7 +45,7 @@
</PageHeader> </PageHeader>
<!-- Main Body of the Wizard --> <!-- Main Body of the Wizard -->
<div class="flex text-base w-full flex-col" v-if="!complete"> <div class="flex text-base w-full flex-col">
<!-- Select Import Type --> <!-- Select Import Type -->
<div <div
class=" class="
@ -367,16 +367,16 @@
</template> </template>
<script lang="ts"> <script lang="ts">
import { DocValue } from 'fyo/core/types'; import { DocValue } from 'fyo/core/types';
import { RawValue } from 'schemas/types';
import { Action as BaseAction } from 'fyo/model/types'; import { Action as BaseAction } from 'fyo/model/types';
import { ValidationError } from 'fyo/utils/errors'; import { ValidationError } from 'fyo/utils/errors';
import { ModelNameEnum } from 'models/types'; import { ModelNameEnum } from 'models/types';
import { OptionField, SelectOption } from 'schemas/types'; import { OptionField, RawValue, SelectOption } from 'schemas/types';
import Button from 'src/components/Button.vue'; import Button from 'src/components/Button.vue';
import AutoComplete from 'src/components/Controls/AutoComplete.vue'; import AutoComplete from 'src/components/Controls/AutoComplete.vue';
import Check from 'src/components/Controls/Check.vue'; import Check from 'src/components/Controls/Check.vue';
import Data from 'src/components/Controls/Data.vue'; import Data from 'src/components/Controls/Data.vue';
import FormControl from 'src/components/Controls/FormControl.vue'; import FormControl from 'src/components/Controls/FormControl.vue';
import Select from 'src/components/Controls/Select.vue';
import DropdownWithActions from 'src/components/DropdownWithActions.vue'; import DropdownWithActions from 'src/components/DropdownWithActions.vue';
import FormHeader from 'src/components/FormHeader.vue'; import FormHeader from 'src/components/FormHeader.vue';
import Modal from 'src/components/Modal.vue'; import Modal from 'src/components/Modal.vue';
@ -389,8 +389,6 @@ import { docsPathRef } from 'src/utils/refs';
import { showMessageDialog } from 'src/utils/ui'; import { showMessageDialog } from 'src/utils/ui';
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import Loading from '../components/Loading.vue'; import Loading from '../components/Loading.vue';
import Select from 'src/components/Controls/Select.vue';
import { isWeakMap } from 'lodash';
type Action = Pick<BaseAction, 'condition' | 'component'> & { type Action = Pick<BaseAction, 'condition' | 'component'> & {
action: Function; action: Function;
@ -443,7 +441,6 @@ export default defineComponent({
if (fyo.store.isDevelopment) { if (fyo.store.isDevelopment) {
// @ts-ignore // @ts-ignore
window.iw = this; window.iw = this;
this.setImportType('Item');
} }
}, },
watch: { watch: {
@ -532,14 +529,13 @@ export default defineComponent({
return this.numColumnsPicked; return this.numColumnsPicked;
} }
let vmColumnCount = 0; if (!this.importer.valueMatrix.length) {
if (this.importer.valueMatrix.length) { return this.importer.assignedTemplateFields.length;
vmColumnCount = this.importer.valueMatrix[0].length;
} }
return Math.min( return Math.min(
this.importer.assignedTemplateFields.length, this.importer.assignedTemplateFields.length,
vmColumnCount this.importer.valueMatrix[0].length
); );
}, },
columnIterator(): number[] { columnIterator(): number[] {
@ -587,6 +583,10 @@ export default defineComponent({
ModelNameEnum.Party, ModelNameEnum.Party,
ModelNameEnum.Item, ModelNameEnum.Item,
ModelNameEnum.JournalEntry, ModelNameEnum.JournalEntry,
ModelNameEnum.Tax,
ModelNameEnum.Account,
ModelNameEnum.Address,
ModelNameEnum.NumberSeries,
]; ];
const hasInventory = fyo.doc.singles.AccountingSettings?.enableInventory; const hasInventory = fyo.doc.singles.AccountingSettings?.enableInventory;

View File

@ -6,7 +6,7 @@ export function getValueMapFromList<T, K extends keyof T, V extends keyof T>(
key: K, key: K,
valueKey: V, valueKey: V,
filterUndefined: boolean = true filterUndefined: boolean = true
): Record<string, unknown> { ): Record<string, T[V]> {
if (filterUndefined) { if (filterUndefined) {
list = list.filter( list = list.filter(
(f) => (f) =>
@ -20,7 +20,7 @@ export function getValueMapFromList<T, K extends keyof T, V extends keyof T>(
const value = f[valueKey]; const value = f[valueKey];
acc[keyValue] = value; acc[keyValue] = value;
return acc; return acc;
}, {} as Record<string, unknown>); }, {} as Record<string, T[V]>);
} }
export function getRandomString(): string { export function getRandomString(): string {