2
0
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:
18alantom 2023-03-07 12:39:49 +05:30
parent f9fbefad0e
commit dabcbcd2ce
10 changed files with 91 additions and 83 deletions

View File

@ -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);
} }

View File

@ -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;

View File

@ -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));

View File

@ -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';

View File

@ -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',
],
}; };
} }

View File

@ -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';

View File

@ -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>`,
};
}, },
}, },
], ],

View File

@ -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';

View File

@ -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>

View File

@ -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 {