mirror of
https://github.com/frappe/books.git
synced 2025-01-03 15:17:30 +00:00
fix: PrintTemplate list view
- type ListCell - improve ListViewConfig types, update where required
This commit is contained in:
parent
f9fbefad0e
commit
dabcbcd2ce
@ -13,7 +13,7 @@ import { TelemetryManager } from './telemetry/telemetry';
|
|||||||
import {
|
import {
|
||||||
DEFAULT_CURRENCY,
|
DEFAULT_CURRENCY,
|
||||||
DEFAULT_DISPLAY_PRECISION,
|
DEFAULT_DISPLAY_PRECISION,
|
||||||
DEFAULT_INTERNAL_PRECISION
|
DEFAULT_INTERNAL_PRECISION,
|
||||||
} from './utils/consts';
|
} from './utils/consts';
|
||||||
import * as errors from './utils/errors';
|
import * as errors from './utils/errors';
|
||||||
import { format } from './utils/format';
|
import { format } from './utils/format';
|
||||||
@ -88,7 +88,7 @@ export class Fyo {
|
|||||||
return this.db.fieldMap;
|
return this.db.fieldMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
format(value: DocValue, field: string | Field, doc?: Doc) {
|
format(value: unknown, field: string | Field, doc?: Doc) {
|
||||||
return format(value, field, doc ?? null, this);
|
return format(value, field, doc ?? null, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import { Fyo } from 'fyo';
|
||||||
import { DocValue, DocValueMap } from 'fyo/core/types';
|
import { DocValue, DocValueMap } from 'fyo/core/types';
|
||||||
import SystemSettings from 'fyo/models/SystemSettings';
|
import SystemSettings from 'fyo/models/SystemSettings';
|
||||||
import { FieldType, Schema, SelectOption } from 'schemas/types';
|
import { FieldType, Schema, SelectOption } from 'schemas/types';
|
||||||
@ -76,13 +77,12 @@ export interface RenderData {
|
|||||||
[key: string]: DocValue | Schema
|
[key: string]: DocValue | Schema
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ColumnConfig {
|
export type ColumnConfig = {
|
||||||
label: string;
|
label: string;
|
||||||
fieldtype: FieldType;
|
fieldtype: FieldType;
|
||||||
fieldname?: string;
|
fieldname: string;
|
||||||
size?: string;
|
|
||||||
render?: (doc: RenderData) => { template: string };
|
render?: (doc: RenderData) => { template: string };
|
||||||
getValue?: (doc: Doc) => string;
|
display?: (value: unknown, fyo: Fyo) => string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ListViewColumn = string | ColumnConfig;
|
export type ListViewColumn = string | ColumnConfig;
|
||||||
|
@ -1,10 +1,9 @@
|
|||||||
import { Fyo } from 'fyo';
|
import { Fyo } from 'fyo';
|
||||||
import { DocValue } from 'fyo/core/types';
|
|
||||||
import { Doc } from 'fyo/model/doc';
|
import { Doc } from 'fyo/model/doc';
|
||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
import { Money } from 'pesa';
|
import { Money } from 'pesa';
|
||||||
import { Field, FieldType, FieldTypeEnum } from 'schemas/types';
|
import { Field, FieldType, FieldTypeEnum } from 'schemas/types';
|
||||||
import { getIsNullOrUndef, safeParseFloat } from 'utils';
|
import { getIsNullOrUndef, safeParseFloat, titleCase } from 'utils';
|
||||||
import {
|
import {
|
||||||
DEFAULT_CURRENCY,
|
DEFAULT_CURRENCY,
|
||||||
DEFAULT_DATE_FORMAT,
|
DEFAULT_DATE_FORMAT,
|
||||||
@ -13,7 +12,7 @@ import {
|
|||||||
} from './consts';
|
} from './consts';
|
||||||
|
|
||||||
export function format(
|
export function format(
|
||||||
value: DocValue,
|
value: unknown,
|
||||||
df: string | Field | null,
|
df: string | Field | null,
|
||||||
doc: Doc | null,
|
doc: Doc | null,
|
||||||
fyo: Fyo
|
fyo: Fyo
|
||||||
@ -45,7 +44,7 @@ export function format(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (field.fieldtype === FieldTypeEnum.Check) {
|
if (field.fieldtype === FieldTypeEnum.Check) {
|
||||||
return Boolean(value).toString();
|
return titleCase(Boolean(value).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getIsNullOrUndef(value)) {
|
if (getIsNullOrUndef(value)) {
|
||||||
@ -55,26 +54,31 @@ export function format(
|
|||||||
return String(value);
|
return String(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
function toDatetime(value: DocValue) {
|
function toDatetime(value: unknown): DateTime | null {
|
||||||
if (typeof value === 'string') {
|
if (typeof value === 'string') {
|
||||||
return DateTime.fromISO(value);
|
return DateTime.fromISO(value);
|
||||||
} else if (value instanceof Date) {
|
} else if (value instanceof Date) {
|
||||||
return DateTime.fromJSDate(value);
|
return DateTime.fromJSDate(value);
|
||||||
} else {
|
} else if (typeof value === 'number') {
|
||||||
return DateTime.fromSeconds(value as number);
|
return DateTime.fromSeconds(value as number);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatDatetime(value: DocValue, fyo: Fyo): string {
|
function formatDatetime(value: unknown, fyo: Fyo): string {
|
||||||
if (value == null) {
|
if (value == null) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
const dateFormat =
|
const dateFormat =
|
||||||
(fyo.singles.SystemSettings?.dateFormat as string) ?? DEFAULT_DATE_FORMAT;
|
(fyo.singles.SystemSettings?.dateFormat as string) ?? DEFAULT_DATE_FORMAT;
|
||||||
const formattedDatetime = toDatetime(value).toFormat(
|
const dateTime = toDatetime(value);
|
||||||
`${dateFormat} HH:mm:ss`
|
if (!dateTime) {
|
||||||
);
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
const formattedDatetime = dateTime.toFormat(`${dateFormat} HH:mm:ss`);
|
||||||
|
|
||||||
if (value === 'Invalid DateTime') {
|
if (value === 'Invalid DateTime') {
|
||||||
return '';
|
return '';
|
||||||
@ -83,7 +87,7 @@ function formatDatetime(value: DocValue, fyo: Fyo): string {
|
|||||||
return formattedDatetime;
|
return formattedDatetime;
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatDate(value: DocValue, fyo: Fyo): string {
|
function formatDate(value: unknown, fyo: Fyo): string {
|
||||||
if (value == null) {
|
if (value == null) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
@ -91,9 +95,12 @@ function formatDate(value: DocValue, fyo: Fyo): string {
|
|||||||
const dateFormat =
|
const dateFormat =
|
||||||
(fyo.singles.SystemSettings?.dateFormat as string) ?? DEFAULT_DATE_FORMAT;
|
(fyo.singles.SystemSettings?.dateFormat as string) ?? DEFAULT_DATE_FORMAT;
|
||||||
|
|
||||||
const dateValue: DateTime = toDatetime(value);
|
const dateTime = toDatetime(value);
|
||||||
|
if (!dateTime) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
const formattedDate = dateValue.toFormat(dateFormat);
|
const formattedDate = dateTime.toFormat(dateFormat);
|
||||||
if (value === 'Invalid DateTime') {
|
if (value === 'Invalid DateTime') {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
@ -102,7 +109,7 @@ function formatDate(value: DocValue, fyo: Fyo): string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function formatCurrency(
|
function formatCurrency(
|
||||||
value: DocValue,
|
value: unknown,
|
||||||
field: Field,
|
field: Field,
|
||||||
doc: Doc | null,
|
doc: Doc | null,
|
||||||
fyo: Fyo
|
fyo: Fyo
|
||||||
@ -125,7 +132,7 @@ function formatCurrency(
|
|||||||
return valueString;
|
return valueString;
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatNumber(value: DocValue, fyo: Fyo): string {
|
function formatNumber(value: unknown, fyo: Fyo): string {
|
||||||
const numberFormatter = getNumberFormatter(fyo);
|
const numberFormatter = getNumberFormatter(fyo);
|
||||||
if (typeof value === 'number') {
|
if (typeof value === 'number') {
|
||||||
value = fyo.pesa(value.toFixed(20));
|
value = fyo.pesa(value.toFixed(20));
|
||||||
|
@ -70,8 +70,8 @@ export class JournalEntry extends Transactional {
|
|||||||
'name',
|
'name',
|
||||||
{
|
{
|
||||||
label: t`Status`,
|
label: t`Status`,
|
||||||
|
fieldname: 'status',
|
||||||
fieldtype: 'Select',
|
fieldtype: 'Select',
|
||||||
size: 'small',
|
|
||||||
render(doc) {
|
render(doc) {
|
||||||
const status = getDocStatus(doc);
|
const status = getDocStatus(doc);
|
||||||
const color = statusColor[status] ?? 'gray';
|
const color = statusColor[status] ?? 'gray';
|
||||||
|
@ -2,6 +2,7 @@ import { Doc } from 'fyo/model/doc';
|
|||||||
import { SchemaMap } from 'schemas/types';
|
import { SchemaMap } from 'schemas/types';
|
||||||
import { ListsMap, ListViewSettings, ReadOnlyMap } from 'fyo/model/types';
|
import { ListsMap, ListViewSettings, ReadOnlyMap } from 'fyo/model/types';
|
||||||
import { ModelNameEnum } from 'models/types';
|
import { ModelNameEnum } from 'models/types';
|
||||||
|
import { Fyo } from 'fyo';
|
||||||
|
|
||||||
export class PrintTemplate extends Doc {
|
export class PrintTemplate extends Doc {
|
||||||
name?: string;
|
name?: string;
|
||||||
@ -17,10 +18,21 @@ export class PrintTemplate extends Doc {
|
|||||||
return super.canDelete;
|
return super.canDelete;
|
||||||
}
|
}
|
||||||
|
|
||||||
static getListViewSettings(): ListViewSettings {
|
static getListViewSettings(fyo: Fyo): ListViewSettings {
|
||||||
return {
|
return {
|
||||||
formRoute: (name) => `/template-builder/${name}`,
|
formRoute: (name) => `/template-builder/${name}`,
|
||||||
columns: ['name', 'type', 'isCustom'],
|
columns: [
|
||||||
|
'name',
|
||||||
|
{
|
||||||
|
label: fyo.t`Type`,
|
||||||
|
fieldtype: 'AutoComplete',
|
||||||
|
fieldname: 'type',
|
||||||
|
display(value) {
|
||||||
|
return fyo.schemaMap[value as string]?.label ?? '';
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'isCustom',
|
||||||
|
],
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -311,7 +311,6 @@ export function getDocStatusListColumn(): ColumnConfig {
|
|||||||
label: t`Status`,
|
label: t`Status`,
|
||||||
fieldname: 'status',
|
fieldname: 'status',
|
||||||
fieldtype: 'Select',
|
fieldtype: 'Select',
|
||||||
size: 'small',
|
|
||||||
render(doc) {
|
render(doc) {
|
||||||
const status = getDocStatus(doc);
|
const status = getDocStatus(doc);
|
||||||
const color = statusColor[status] ?? 'gray';
|
const color = statusColor[status] ?? 'gray';
|
||||||
|
@ -80,6 +80,13 @@ export class StockMovement extends Transfer {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static getListViewSettings(fyo: Fyo): ListViewSettings {
|
static getListViewSettings(fyo: Fyo): ListViewSettings {
|
||||||
|
const movementTypeMap = {
|
||||||
|
[MovementType.MaterialIssue]: fyo.t`Material Issue`,
|
||||||
|
[MovementType.MaterialReceipt]: fyo.t`Material Receipt`,
|
||||||
|
[MovementType.MaterialTransfer]: fyo.t`Material Transfer`,
|
||||||
|
[MovementType.Manufacture]: fyo.t`Manufacture`,
|
||||||
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
formRoute: (name) => `/edit/StockMovement/${name}`,
|
formRoute: (name) => `/edit/StockMovement/${name}`,
|
||||||
columns: [
|
columns: [
|
||||||
@ -90,20 +97,8 @@ export class StockMovement extends Transfer {
|
|||||||
label: fyo.t`Movement Type`,
|
label: fyo.t`Movement Type`,
|
||||||
fieldname: 'movementType',
|
fieldname: 'movementType',
|
||||||
fieldtype: 'Select',
|
fieldtype: 'Select',
|
||||||
size: 'small',
|
display(value): string {
|
||||||
render(doc) {
|
return movementTypeMap[value as MovementType] ?? '';
|
||||||
const movementType = doc.movementType as MovementType;
|
|
||||||
const label =
|
|
||||||
{
|
|
||||||
[MovementType.MaterialIssue]: fyo.t`Material Issue`,
|
|
||||||
[MovementType.MaterialReceipt]: fyo.t`Material Receipt`,
|
|
||||||
[MovementType.MaterialTransfer]: fyo.t`Material Transfer`,
|
|
||||||
[MovementType.Manufacture]: fyo.t`Manufacture`,
|
|
||||||
}[movementType] ?? '';
|
|
||||||
|
|
||||||
return {
|
|
||||||
template: `<span>${label}</span>`,
|
|
||||||
};
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -37,7 +37,7 @@
|
|||||||
|
|
||||||
<!-- Data Rows -->
|
<!-- Data Rows -->
|
||||||
<div class="overflow-y-auto custom-scroll" v-if="dataSlice.length !== 0">
|
<div class="overflow-y-auto custom-scroll" v-if="dataSlice.length !== 0">
|
||||||
<div v-for="(doc, i) in dataSlice" :key="doc.name">
|
<div v-for="(row, i) in dataSlice" :key="row.name">
|
||||||
<!-- Row Content -->
|
<!-- Row Content -->
|
||||||
<div class="flex hover:bg-gray-50 items-center">
|
<div class="flex hover:bg-gray-50 items-center">
|
||||||
<p class="w-8 text-end me-4 text-gray-900">
|
<p class="w-8 text-end me-4 text-gray-900">
|
||||||
@ -46,7 +46,7 @@
|
|||||||
<Row
|
<Row
|
||||||
gap="1rem"
|
gap="1rem"
|
||||||
class="cursor-pointer text-gray-900 flex-1 h-row-mid"
|
class="cursor-pointer text-gray-900 flex-1 h-row-mid"
|
||||||
@click="$emit('openDoc', doc.name)"
|
@click="$emit('openDoc', row.name)"
|
||||||
:columnCount="columns.length"
|
:columnCount="columns.length"
|
||||||
>
|
>
|
||||||
<ListCell
|
<ListCell
|
||||||
@ -56,7 +56,7 @@
|
|||||||
'text-end': isNumeric(column.fieldtype),
|
'text-end': isNumeric(column.fieldtype),
|
||||||
'pe-4': c === columns.length - 1,
|
'pe-4': c === columns.length - 1,
|
||||||
}"
|
}"
|
||||||
:doc="doc"
|
:row="row"
|
||||||
:column="column"
|
:column="column"
|
||||||
/>
|
/>
|
||||||
</Row>
|
</Row>
|
||||||
@ -95,7 +95,6 @@ import Paginator from 'src/components/Paginator.vue';
|
|||||||
import Row from 'src/components/Row';
|
import Row from 'src/components/Row';
|
||||||
import { fyo } from 'src/initFyo';
|
import { fyo } from 'src/initFyo';
|
||||||
import { isNumeric } from 'src/utils';
|
import { isNumeric } from 'src/utils';
|
||||||
import { openQuickEdit, routeTo } from 'src/utils/ui';
|
|
||||||
import { objectForEach } from 'utils/index';
|
import { objectForEach } from 'utils/index';
|
||||||
import { defineComponent, toRaw } from 'vue';
|
import { defineComponent, toRaw } from 'vue';
|
||||||
import ListCell from './ListCell';
|
import ListCell from './ListCell';
|
||||||
|
@ -4,26 +4,52 @@
|
|||||||
<component v-else :is="customRenderer" />
|
<component v-else :is="customRenderer" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<script>
|
<script lang="ts">
|
||||||
|
import { ColumnConfig, RenderData } from 'fyo/model/types';
|
||||||
|
import { Field } from 'schemas/types';
|
||||||
import { fyo } from 'src/initFyo';
|
import { fyo } from 'src/initFyo';
|
||||||
import { isNumeric } from 'src/utils';
|
import { isNumeric } from 'src/utils';
|
||||||
|
import { defineComponent, PropType } from 'vue';
|
||||||
|
|
||||||
export default {
|
type Column = ColumnConfig | Field;
|
||||||
|
|
||||||
|
function isField(column: ColumnConfig | Field): column is Field {
|
||||||
|
if ((column as ColumnConfig).display || (column as ColumnConfig).render) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
name: 'ListCell',
|
name: 'ListCell',
|
||||||
props: ['doc', 'column'],
|
props: {
|
||||||
|
row: { type: Object as PropType<RenderData>, required: true },
|
||||||
|
column: { type: Object as PropType<Column>, required: true },
|
||||||
|
},
|
||||||
computed: {
|
computed: {
|
||||||
columnValue() {
|
columnValue(): string {
|
||||||
let { column, doc } = this;
|
const column = this.column;
|
||||||
let value = doc[column.fieldname];
|
const value = this.row[this.column.fieldname];
|
||||||
return fyo.format(value, column, doc);
|
|
||||||
|
if (isField(column)) {
|
||||||
|
return fyo.format(value, column);
|
||||||
|
}
|
||||||
|
|
||||||
|
return column.display?.(value, fyo) ?? '';
|
||||||
},
|
},
|
||||||
customRenderer() {
|
customRenderer() {
|
||||||
if (!this.column.render) return;
|
const { render } = this.column as ColumnConfig;
|
||||||
return this.column.render(this.doc);
|
|
||||||
|
if (!render) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return render(this.row);
|
||||||
},
|
},
|
||||||
cellClass() {
|
cellClass() {
|
||||||
return isNumeric(this.column.fieldtype) ? 'justify-end' : '';
|
return isNumeric(this.column.fieldtype) ? 'justify-end' : '';
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
});
|
||||||
</script>
|
</script>
|
||||||
|
@ -344,36 +344,6 @@ async function openEdit({ name, schemaName }: Doc) {
|
|||||||
|
|
||||||
const route = getFormRoute(schemaName, name);
|
const route = getFormRoute(schemaName, name);
|
||||||
return await routeTo(route);
|
return await routeTo(route);
|
||||||
|
|
||||||
/*
|
|
||||||
const listConfig = fyo.models[doc.schemaName]?.getListViewSettings?.(fyo);
|
|
||||||
const formRoute = listConfig?.formRoute;
|
|
||||||
if (!doc.name) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (formRoute) {
|
|
||||||
const route = formRoute(doc.name);
|
|
||||||
return await routeTo(route);
|
|
||||||
}
|
|
||||||
|
|
||||||
const isFormEdit = [
|
|
||||||
ModelNameEnum.SalesInvoice,
|
|
||||||
ModelNameEnum.PurchaseInvoice,
|
|
||||||
ModelNameEnum.JournalEntry,
|
|
||||||
ModelNameEnum.Shipment,
|
|
||||||
ModelNameEnum.PurchaseReceipt,
|
|
||||||
ModelNameEnum.StockMovement,
|
|
||||||
ModelNameEnum.Payment,
|
|
||||||
ModelNameEnum.Item,
|
|
||||||
].includes(doc.schemaName as ModelNameEnum);
|
|
||||||
|
|
||||||
if (isFormEdit) {
|
|
||||||
return await routeTo(`/edit/${doc.schemaName}/${doc.name}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
await openQuickEdit({ schemaName: doc.schemaName, name: doc.name });
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDuplicateAction(doc: Doc): Action {
|
function getDuplicateAction(doc: Doc): Action {
|
||||||
|
Loading…
Reference in New Issue
Block a user