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:
parent
aa04e1493c
commit
48797dff62
@ -73,7 +73,6 @@
|
|||||||
"label": "Address",
|
"label": "Address",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"target": "Address",
|
"target": "Address",
|
||||||
"placeholder": "Click to create",
|
|
||||||
"inline": true
|
"inline": true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -40,7 +40,6 @@
|
|||||||
"label": "Address",
|
"label": "Address",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"target": "Address",
|
"target": "Address",
|
||||||
"placeholder": "Click to create",
|
|
||||||
"inline": true
|
"inline": true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
"label": "Address",
|
"label": "Address",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"target": "Address",
|
"target": "Address",
|
||||||
"placeholder": "Click to create",
|
|
||||||
"inline": true
|
"inline": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
@ -73,7 +73,6 @@
|
|||||||
"label": "Address",
|
"label": "Address",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"target": "Address",
|
"target": "Address",
|
||||||
"placeholder": "Click to create",
|
|
||||||
"inline": true
|
"inline": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
@ -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}`;
|
||||||
|
@ -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;
|
||||||
|
@ -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 {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user