2019-10-04 20:18:10 +00:00
|
|
|
<template>
|
2022-06-09 12:27:11 +00:00
|
|
|
<div
|
2023-01-07 13:02:23 +00:00
|
|
|
class="border-s h-full overflow-auto w-quick-edit"
|
2022-06-09 12:27:11 +00:00
|
|
|
:class="white ? 'bg-white' : 'bg-gray-25'"
|
|
|
|
>
|
2022-04-28 12:37:07 +00:00
|
|
|
<!-- Quick edit Tool bar -->
|
2022-07-11 08:45:37 +00:00
|
|
|
<div
|
|
|
|
class="flex items-center justify-between px-4 h-row-largest"
|
2022-07-12 10:23:41 +00:00
|
|
|
:class="{ 'border-b': showName }"
|
2022-07-11 08:45:37 +00:00
|
|
|
>
|
2022-04-28 12:37:07 +00:00
|
|
|
<!-- Close Button and Status Text -->
|
2019-11-19 19:08:49 +00:00
|
|
|
<div class="flex items-center">
|
2019-12-03 13:10:21 +00:00
|
|
|
<Button :icon="true" @click="routeToPrevious">
|
2019-11-19 19:08:49 +00:00
|
|
|
<feather-icon name="x" class="w-4 h-4" />
|
|
|
|
</Button>
|
2023-01-07 10:55:31 +00:00
|
|
|
<span v-if="statusText" class="ms-2 text-base text-gray-600">{{
|
2019-11-19 19:08:49 +00:00
|
|
|
statusText
|
|
|
|
}}</span>
|
|
|
|
</div>
|
2022-04-28 12:37:07 +00:00
|
|
|
|
|
|
|
<!-- Actions, Badge and Status Change Buttons -->
|
2022-07-12 10:23:41 +00:00
|
|
|
<div class="flex items-stretch gap-2" v-if="showSave">
|
2022-07-11 10:52:04 +00:00
|
|
|
<StatusBadge :status="status" />
|
|
|
|
<DropdownWithActions :actions="actions" />
|
2019-11-19 19:08:49 +00:00
|
|
|
<Button
|
|
|
|
:icon="true"
|
2022-04-24 06:48:44 +00:00
|
|
|
@click="sync"
|
2019-11-19 19:08:49 +00:00
|
|
|
type="primary"
|
2022-12-14 10:40:02 +00:00
|
|
|
v-if="doc?.canSave"
|
2022-06-14 09:10:46 +00:00
|
|
|
class="text-white text-xs"
|
2019-11-19 19:08:49 +00:00
|
|
|
>
|
2022-02-14 05:15:26 +00:00
|
|
|
{{ t`Save` }}
|
2019-11-19 19:08:49 +00:00
|
|
|
</Button>
|
|
|
|
<Button
|
|
|
|
:icon="true"
|
2022-04-29 13:00:24 +00:00
|
|
|
@click="submit"
|
2019-11-19 19:08:49 +00:00
|
|
|
type="primary"
|
2022-12-14 10:40:02 +00:00
|
|
|
v-else-if="doc?.canSubmit"
|
2022-06-14 09:10:46 +00:00
|
|
|
class="text-white text-xs"
|
2019-11-19 19:08:49 +00:00
|
|
|
>
|
2022-02-14 05:15:26 +00:00
|
|
|
{{ t`Submit` }}
|
2019-11-19 19:08:49 +00:00
|
|
|
</Button>
|
|
|
|
</div>
|
2019-10-04 20:18:10 +00:00
|
|
|
</div>
|
2022-04-28 12:37:07 +00:00
|
|
|
|
|
|
|
<!-- Name and image -->
|
2022-05-31 12:15:25 +00:00
|
|
|
<div
|
2022-10-25 08:54:33 +00:00
|
|
|
class="items-center"
|
|
|
|
:class="imageField ? 'grid' : 'flex justify-center'"
|
|
|
|
:style="{
|
|
|
|
height: `calc(var(--h-row-mid) * ${!!imageField ? '2 + 1px' : '1'})`,
|
|
|
|
gridTemplateColumns: `minmax(0, 1.1fr) minmax(0, 2fr)`,
|
|
|
|
}"
|
|
|
|
v-if="doc && showName && (titleField || imageField)"
|
2022-05-31 12:15:25 +00:00
|
|
|
>
|
2022-04-29 13:00:24 +00:00
|
|
|
<FormControl
|
|
|
|
v-if="imageField"
|
2023-01-07 10:55:31 +00:00
|
|
|
class="ms-4"
|
2022-04-29 13:00:24 +00:00
|
|
|
:df="imageField"
|
|
|
|
:value="doc[imageField.fieldname]"
|
|
|
|
@change="(value) => valueChange(imageField, value)"
|
2022-10-25 08:01:49 +00:00
|
|
|
:letter-placeholder="doc[titleField.fieldname]?.[0] ?? ''"
|
2022-04-29 13:00:24 +00:00
|
|
|
/>
|
|
|
|
<FormControl
|
|
|
|
v-if="titleField"
|
2023-01-07 10:55:31 +00:00
|
|
|
:class="!!imageField ? 'me-4' : 'w-full mx-4'"
|
2022-10-25 08:54:33 +00:00
|
|
|
:input-class="[
|
|
|
|
'font-semibold text-xl',
|
|
|
|
!!imageField ? '' : 'text-center',
|
|
|
|
]"
|
|
|
|
ref="titleControl"
|
|
|
|
size="small"
|
2022-04-29 13:00:24 +00:00
|
|
|
:df="titleField"
|
|
|
|
:value="doc[titleField.fieldname]"
|
2022-10-25 08:54:33 +00:00
|
|
|
:read-only="doc.inserted || doc.schema.naming !== 'manual'"
|
2022-04-29 13:00:24 +00:00
|
|
|
@change="(value) => valueChange(titleField, value)"
|
|
|
|
/>
|
2019-10-04 20:18:10 +00:00
|
|
|
</div>
|
2022-04-28 12:37:07 +00:00
|
|
|
|
|
|
|
<!-- Rest of the form -->
|
2019-11-19 19:08:49 +00:00
|
|
|
<TwoColumnForm
|
2022-10-13 11:58:53 +00:00
|
|
|
class="w-full"
|
2019-11-19 19:08:49 +00:00
|
|
|
ref="form"
|
|
|
|
v-if="doc"
|
|
|
|
:doc="doc"
|
|
|
|
:fields="fields"
|
2022-04-28 12:37:07 +00:00
|
|
|
:autosave="false"
|
2019-11-19 19:08:49 +00:00
|
|
|
:column-ratio="[1.1, 2]"
|
|
|
|
/>
|
2022-04-28 12:37:07 +00:00
|
|
|
|
|
|
|
<!-- QuickEdit Widgets -->
|
|
|
|
<component v-if="quickEditWidget" :is="quickEditWidget" />
|
2019-10-04 20:18:10 +00:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2022-04-29 19:04:08 +00:00
|
|
|
import { computed } from '@vue/reactivity';
|
2022-04-21 13:08:36 +00:00
|
|
|
import { t } from 'fyo';
|
2022-07-11 10:52:04 +00:00
|
|
|
import { Doc } from 'fyo/model/doc';
|
2022-06-14 09:10:46 +00:00
|
|
|
import { getDocStatus } from 'models/helpers';
|
2022-04-28 06:34:55 +00:00
|
|
|
import Button from 'src/components/Button.vue';
|
2022-04-22 11:02:03 +00:00
|
|
|
import FormControl from 'src/components/Controls/FormControl.vue';
|
2022-04-28 06:34:55 +00:00
|
|
|
import DropdownWithActions from 'src/components/DropdownWithActions.vue';
|
|
|
|
import StatusBadge from 'src/components/StatusBadge.vue';
|
|
|
|
import TwoColumnForm from 'src/components/TwoColumnForm.vue';
|
2022-04-21 13:08:36 +00:00
|
|
|
import { fyo } from 'src/initFyo';
|
2022-04-28 06:34:55 +00:00
|
|
|
import { getQuickEditWidget } from 'src/utils/quickEditWidgets';
|
2022-11-30 13:35:49 +00:00
|
|
|
import { getActionsForDoc, openQuickEdit } from 'src/utils/ui';
|
2019-10-04 20:18:10 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
name: 'QuickEditForm',
|
2022-04-28 06:34:55 +00:00
|
|
|
props: {
|
|
|
|
name: String,
|
|
|
|
schemaName: String,
|
|
|
|
defaults: String,
|
2022-06-09 12:27:11 +00:00
|
|
|
white: { type: Boolean, default: false },
|
2022-07-11 10:52:04 +00:00
|
|
|
routeBack: { type: Boolean, default: true },
|
2022-07-12 10:23:41 +00:00
|
|
|
showName: { type: Boolean, default: true },
|
|
|
|
showSave: { type: Boolean, default: true },
|
|
|
|
sourceDoc: { type: Doc, default: null },
|
|
|
|
loadOnClose: { type: Boolean, default: true },
|
|
|
|
sourceFields: { type: Array, default: () => [] },
|
2022-04-28 06:34:55 +00:00
|
|
|
hideFields: { type: Array, default: () => [] },
|
|
|
|
showFields: { type: Array, default: () => [] },
|
|
|
|
},
|
2019-10-04 20:18:10 +00:00
|
|
|
components: {
|
|
|
|
Button,
|
2019-11-19 19:08:49 +00:00
|
|
|
FormControl,
|
2021-11-11 09:35:34 +00:00
|
|
|
StatusBadge,
|
2019-11-19 19:08:49 +00:00
|
|
|
TwoColumnForm,
|
2021-11-08 14:07:06 +00:00
|
|
|
DropdownWithActions,
|
2019-10-04 20:18:10 +00:00
|
|
|
},
|
2022-07-11 10:52:04 +00:00
|
|
|
emits: ['close'],
|
2023-01-20 11:01:09 +00:00
|
|
|
inject: ['shortcuts'],
|
2019-10-06 12:33:21 +00:00
|
|
|
provide() {
|
|
|
|
return {
|
2022-04-28 06:34:55 +00:00
|
|
|
schemaName: this.schemaName,
|
2019-11-19 19:08:49 +00:00
|
|
|
name: this.name,
|
2022-04-29 19:04:08 +00:00
|
|
|
doc: computed(() => this.doc),
|
2019-10-11 09:55:50 +00:00
|
|
|
};
|
2019-10-06 12:33:21 +00:00
|
|
|
},
|
2019-10-04 20:18:10 +00:00
|
|
|
data() {
|
|
|
|
return {
|
2019-11-19 19:08:49 +00:00
|
|
|
doc: null,
|
2022-02-11 10:25:49 +00:00
|
|
|
values: null,
|
2019-11-19 19:08:49 +00:00
|
|
|
titleField: null,
|
|
|
|
imageField: null,
|
2021-11-08 14:07:06 +00:00
|
|
|
statusText: null,
|
2019-10-04 20:18:10 +00:00
|
|
|
};
|
|
|
|
},
|
2022-04-29 19:04:08 +00:00
|
|
|
mounted() {
|
|
|
|
if (this.defaults) {
|
|
|
|
this.values = JSON.parse(this.defaults);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fyo.store.isDevelopment) {
|
|
|
|
window.qef = this;
|
|
|
|
}
|
|
|
|
},
|
2019-11-19 19:08:49 +00:00
|
|
|
async created() {
|
2022-04-28 06:34:55 +00:00
|
|
|
await this.fetchFieldsAndDoc();
|
2019-10-05 21:41:16 +00:00
|
|
|
},
|
2019-11-19 19:08:49 +00:00
|
|
|
computed: {
|
2022-07-11 08:45:37 +00:00
|
|
|
isChild() {
|
|
|
|
return !!this?.doc?.schema?.isChild;
|
|
|
|
},
|
2022-04-28 06:34:55 +00:00
|
|
|
schema() {
|
|
|
|
return fyo.schemaMap[this.schemaName] ?? null;
|
2019-11-19 19:08:49 +00:00
|
|
|
},
|
2021-11-11 09:35:34 +00:00
|
|
|
status() {
|
2022-06-14 09:10:46 +00:00
|
|
|
return getDocStatus(this.doc);
|
2021-11-11 09:35:34 +00:00
|
|
|
},
|
2019-11-19 19:08:49 +00:00
|
|
|
fields() {
|
2022-07-12 10:23:41 +00:00
|
|
|
if (this.sourceFields?.length) {
|
|
|
|
return this.sourceFields;
|
|
|
|
}
|
|
|
|
|
2022-04-28 06:34:55 +00:00
|
|
|
if (!this.schema) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
const fieldnames = (this.schema.quickEditFields ?? ['name']).filter(
|
|
|
|
(f) => !this.hideFields.includes(f)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (this.showFields?.length) {
|
|
|
|
fieldnames.push(
|
|
|
|
...this.schema.fields
|
|
|
|
.map((f) => f.fieldname)
|
|
|
|
.filter((f) => this.showFields.includes(f))
|
2022-02-04 06:36:44 +00:00
|
|
|
);
|
|
|
|
}
|
2022-04-28 06:34:55 +00:00
|
|
|
|
|
|
|
return fieldnames.map((f) => fyo.getField(this.schemaName, f));
|
2019-11-19 19:08:49 +00:00
|
|
|
},
|
|
|
|
actions() {
|
2022-11-30 13:35:49 +00:00
|
|
|
return getActionsForDoc(this.doc);
|
2019-12-26 13:45:41 +00:00
|
|
|
},
|
|
|
|
quickEditWidget() {
|
2022-04-28 12:37:07 +00:00
|
|
|
if (this.doc?.notInserted ?? true) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-04-28 06:34:55 +00:00
|
|
|
const widget = getQuickEditWidget(this.schemaName);
|
|
|
|
if (widget === null) {
|
2019-12-26 13:45:41 +00:00
|
|
|
return null;
|
|
|
|
}
|
2022-04-28 06:34:55 +00:00
|
|
|
|
|
|
|
return widget(this.doc);
|
2021-11-08 14:07:06 +00:00
|
|
|
},
|
2019-11-19 19:08:49 +00:00
|
|
|
},
|
|
|
|
methods: {
|
2022-04-28 06:34:55 +00:00
|
|
|
async fetchFieldsAndDoc() {
|
|
|
|
if (!this.schema) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const titleField = this.schema.titleField;
|
|
|
|
this.titleField = fyo.getField(this.schemaName, titleField);
|
|
|
|
this.imageField = fyo.getField(this.schemaName, 'image');
|
|
|
|
|
2019-10-11 09:55:50 +00:00
|
|
|
await this.fetchDoc();
|
|
|
|
|
|
|
|
// setup the title field
|
2022-04-28 06:34:55 +00:00
|
|
|
this.setTitleField();
|
2019-11-19 19:08:49 +00:00
|
|
|
|
|
|
|
// set default values
|
2019-10-11 09:55:50 +00:00
|
|
|
if (this.values) {
|
2022-05-01 07:39:03 +00:00
|
|
|
this.doc?.set(this.values);
|
2019-10-11 09:55:50 +00:00
|
|
|
}
|
|
|
|
},
|
2022-04-28 06:34:55 +00:00
|
|
|
setTitleField() {
|
|
|
|
const { fieldname, readOnly } = this.titleField;
|
2022-05-01 07:39:03 +00:00
|
|
|
if (!this.doc?.notInserted || !this?.doc[fieldname]) {
|
2022-04-28 06:34:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-07-15 08:30:25 +00:00
|
|
|
const isManual = this.schema.naming === 'manual';
|
|
|
|
const isNumberSeries = fyo.getField(this.schemaName, 'numberSeries');
|
|
|
|
if (readOnly && (!this?.doc[fieldname] || isNumberSeries)) {
|
2022-04-28 06:34:55 +00:00
|
|
|
this.doc.set(fieldname, t`New ${this.schema.label}`);
|
2022-07-15 08:10:51 +00:00
|
|
|
}
|
|
|
|
|
2022-07-15 08:30:25 +00:00
|
|
|
if (this?.doc[fieldname] && !isManual) {
|
2022-04-28 06:34:55 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.doc.set(fieldname, '');
|
|
|
|
|
|
|
|
setTimeout(() => {
|
2022-04-28 12:37:07 +00:00
|
|
|
this.$refs.titleControl?.focus();
|
2022-04-28 06:34:55 +00:00
|
|
|
}, 300);
|
|
|
|
},
|
2019-11-19 19:08:49 +00:00
|
|
|
async fetchDoc() {
|
2022-07-11 10:52:04 +00:00
|
|
|
if (this.sourceDoc) {
|
|
|
|
return (this.doc = this.sourceDoc);
|
|
|
|
}
|
|
|
|
|
2022-04-29 13:00:24 +00:00
|
|
|
if (!this.schemaName) {
|
|
|
|
this.$router.back();
|
|
|
|
}
|
2019-11-19 19:08:49 +00:00
|
|
|
|
2022-04-29 13:00:24 +00:00
|
|
|
if (this.name) {
|
|
|
|
try {
|
|
|
|
this.doc = await fyo.doc.getDoc(this.schemaName, this.name);
|
|
|
|
} catch (e) {
|
|
|
|
this.$router.back();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.doc = fyo.doc.getNewDoc(this.schemaName);
|
|
|
|
}
|
2022-04-28 12:37:07 +00:00
|
|
|
|
2022-04-29 13:00:24 +00:00
|
|
|
if (this.doc === null) {
|
|
|
|
return;
|
|
|
|
}
|
2022-04-28 12:37:07 +00:00
|
|
|
|
2022-04-29 13:00:24 +00:00
|
|
|
this.doc.once('afterRename', () => {
|
|
|
|
openQuickEdit({
|
|
|
|
schemaName: this.schemaName,
|
|
|
|
name: this.doc.name,
|
2019-11-19 19:08:49 +00:00
|
|
|
});
|
2022-04-29 13:00:24 +00:00
|
|
|
});
|
2019-10-05 21:41:16 +00:00
|
|
|
},
|
2019-12-04 18:40:46 +00:00
|
|
|
valueChange(df, value) {
|
|
|
|
this.$refs.form.onChange(df, value);
|
|
|
|
},
|
2022-04-29 13:00:24 +00:00
|
|
|
async sync() {
|
|
|
|
this.statusText = t`Saving`;
|
|
|
|
try {
|
|
|
|
await this.$refs.form.sync();
|
|
|
|
setTimeout(() => {
|
|
|
|
this.statusText = null;
|
|
|
|
}, 300);
|
|
|
|
} catch (err) {
|
|
|
|
this.statusText = null;
|
|
|
|
console.error(err);
|
|
|
|
}
|
2019-10-05 21:41:16 +00:00
|
|
|
},
|
2022-04-29 13:00:24 +00:00
|
|
|
async submit() {
|
|
|
|
this.statusText = t`Submitting`;
|
2021-11-08 14:07:06 +00:00
|
|
|
try {
|
|
|
|
await this.$refs.form.submit();
|
2022-04-29 13:00:24 +00:00
|
|
|
setTimeout(() => {
|
|
|
|
this.statusText = null;
|
|
|
|
}, 300);
|
|
|
|
} catch (err) {
|
2021-11-08 14:07:06 +00:00
|
|
|
this.statusText = null;
|
2022-04-29 13:00:24 +00:00
|
|
|
console.error(err);
|
2021-11-08 14:07:06 +00:00
|
|
|
}
|
2019-11-08 19:58:58 +00:00
|
|
|
},
|
2019-12-03 13:10:21 +00:00
|
|
|
routeToPrevious() {
|
2022-07-12 10:23:41 +00:00
|
|
|
if (this.loadOnClose && this.doc.dirty && !this.doc.notInserted) {
|
2022-04-28 12:37:07 +00:00
|
|
|
this.doc.load();
|
|
|
|
}
|
2022-07-11 10:52:04 +00:00
|
|
|
|
|
|
|
if (this.routeBack) {
|
|
|
|
this.$router.back();
|
|
|
|
} else {
|
|
|
|
this.$emit('close');
|
|
|
|
}
|
2019-12-03 13:10:21 +00:00
|
|
|
},
|
2021-11-08 14:07:06 +00:00
|
|
|
},
|
2019-10-04 20:18:10 +00:00
|
|
|
};
|
|
|
|
</script>
|