2
0
mirror of https://github.com/frappe/books.git synced 2025-01-10 18:24:40 +00:00
books/src/pages/QuickEditForm.vue

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

324 lines
8.1 KiB
Vue
Raw Normal View History

2019-10-04 20:18:10 +00:00
<template>
2022-06-09 12:27:11 +00:00
<div
class="border-l h-full overflow-auto w-quick-edit"
2022-06-09 12:27:11 +00:00
:class="white ? 'bg-white' : 'bg-gray-25'"
>
<!-- Quick edit Tool bar -->
2022-07-11 08:45:37 +00:00
<div
class="flex items-center justify-between px-4 h-row-largest"
:class="{ 'border-b': showName }"
2022-07-11 08:45:37 +00:00
>
<!-- Close Button and Status Text -->
<div class="flex items-center">
<Button :icon="true" @click="routeToPrevious">
<feather-icon name="x" class="w-4 h-4" />
</Button>
<span v-if="statusText" class="ml-2 text-base text-gray-600">{{
statusText
}}</span>
</div>
<!-- Actions, Badge and Status Change Buttons -->
<div class="flex items-stretch gap-2" v-if="showSave">
<StatusBadge :status="status" />
<DropdownWithActions :actions="actions" />
<Button
:icon="true"
@click="sync"
type="primary"
v-if="doc?.canSave"
2022-06-14 09:10:46 +00:00
class="text-white text-xs"
>
{{ t`Save` }}
</Button>
<Button
:icon="true"
@click="submit"
type="primary"
v-else-if="doc?.canSubmit"
2022-06-14 09:10:46 +00:00
class="text-white text-xs"
>
{{ t`Submit` }}
</Button>
</div>
2019-10-04 20:18:10 +00:00
</div>
<!-- Name and image -->
<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)"
>
<FormControl
v-if="imageField"
2022-10-25 08:54:33 +00:00
class="ml-4"
:df="imageField"
:value="doc[imageField.fieldname]"
@change="(value) => valueChange(imageField, value)"
:letter-placeholder="doc[titleField.fieldname]?.[0] ?? ''"
/>
<FormControl
v-if="titleField"
2022-10-25 08:58:13 +00:00
:class="!!imageField ? 'mr-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"
:df="titleField"
:value="doc[titleField.fieldname]"
2022-10-25 08:54:33 +00:00
:read-only="doc.inserted || doc.schema.naming !== 'manual'"
@change="(value) => valueChange(titleField, value)"
/>
2019-10-04 20:18:10 +00:00
</div>
<!-- Rest of the form -->
<TwoColumnForm
class="w-full"
ref="form"
v-if="doc"
:doc="doc"
:fields="fields"
:autosave="false"
:column-ratio="[1.1, 2]"
/>
<!-- 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';
import { t } from 'fyo';
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';
import { fyo } from 'src/initFyo';
2022-04-28 06:34:55 +00:00
import { getQuickEditWidget } from 'src/utils/quickEditWidgets';
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 },
routeBack: { type: Boolean, default: true },
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,
FormControl,
StatusBadge,
TwoColumnForm,
DropdownWithActions,
2019-10-04 20:18:10 +00:00
},
emits: ['close'],
2019-10-06 12:33:21 +00:00
provide() {
return {
2022-04-28 06:34:55 +00:00
schemaName: this.schemaName,
name: this.name,
2022-04-29 19:04:08 +00:00
doc: computed(() => this.doc),
};
2019-10-06 12:33:21 +00:00
},
2019-10-04 20:18:10 +00:00
data() {
return {
doc: null,
values: null,
titleField: null,
imageField: null,
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;
}
},
async created() {
2022-04-28 06:34:55 +00:00
await this.fetchFieldsAndDoc();
},
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;
},
status() {
2022-06-14 09:10:46 +00:00
return getDocStatus(this.doc);
},
fields() {
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-04-28 06:34:55 +00:00
return fieldnames.map((f) => fyo.getField(this.schemaName, f));
},
actions() {
return getActionsForDoc(this.doc);
2019-12-26 13:45:41 +00:00
},
quickEditWidget() {
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);
},
},
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');
await this.fetchDoc();
// setup the title field
2022-04-28 06:34:55 +00:00
this.setTitleField();
// set default values
if (this.values) {
this.doc?.set(this.values);
}
},
2022-04-28 06:34:55 +00:00
setTitleField() {
const { fieldname, readOnly } = this.titleField;
if (!this.doc?.notInserted || !this?.doc[fieldname]) {
2022-04-28 06:34:55 +00:00
return;
}
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
}
if (this?.doc[fieldname] && !isManual) {
2022-04-28 06:34:55 +00:00
return;
}
this.doc.set(fieldname, '');
setTimeout(() => {
this.$refs.titleControl?.focus();
2022-04-28 06:34:55 +00:00
}, 300);
},
async fetchDoc() {
if (this.sourceDoc) {
return (this.doc = this.sourceDoc);
}
if (!this.schemaName) {
this.$router.back();
}
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);
}
if (this.doc === null) {
return;
}
this.doc.once('afterRename', () => {
openQuickEdit({
schemaName: this.schemaName,
name: this.doc.name,
});
});
},
2019-12-04 18:40:46 +00:00
valueChange(df, value) {
this.$refs.form.onChange(df, value);
},
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);
}
},
async submit() {
this.statusText = t`Submitting`;
try {
await this.$refs.form.submit();
setTimeout(() => {
this.statusText = null;
}, 300);
} catch (err) {
this.statusText = null;
console.error(err);
}
2019-11-08 19:58:58 +00:00
},
routeToPrevious() {
if (this.loadOnClose && this.doc.dirty && !this.doc.notInserted) {
this.doc.load();
}
if (this.routeBack) {
this.$router.back();
} else {
this.$emit('close');
}
},
},
2019-10-04 20:18:10 +00:00
};
</script>