From b4436316441f68e3781219cc713ccbb980327392 Mon Sep 17 00:00:00 2001
From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com>
Date: Fri, 9 Aug 2024 10:41:54 +0530
Subject: [PATCH 1/4] feat: Lead
---
README.md | 1 -
fyo/model/types.ts | 10 ++
.../AccountingSettings/AccountingSettings.ts | 4 +
models/baseModels/Invoice/Invoice.ts | 4 +
models/baseModels/Lead/Lead.ts | 32 +++++
models/baseModels/Party/Party.ts | 22 +++
models/baseModels/SalesQuote/SalesQuote.ts | 24 +++-
models/baseModels/tests/testLead.spec.ts | 127 ++++++++++++++++++
models/helpers.ts | 107 ++++++++++++++-
models/index.ts | 2 +
models/regionalModels/in/Party.ts | 2 +
models/types.ts | 1 +
schemas/app/AccountingSettings.json | 7 +
schemas/app/Lead.json | 76 +++++++++++
schemas/app/Party.json | 8 ++
schemas/app/SalesQuote.json | 22 ++-
schemas/schemas.ts | 2 +
src/utils/sidebarConfig.ts | 7 +
18 files changed, 452 insertions(+), 6 deletions(-)
create mode 100644 models/baseModels/Lead/Lead.ts
create mode 100644 models/baseModels/tests/testLead.spec.ts
create mode 100644 schemas/app/Lead.json
diff --git a/README.md b/README.md
index f25925d4e..720a6f892 100644
--- a/README.md
+++ b/README.md
@@ -2,7 +2,6 @@
>
> Frappe Books is looking for a maintainer, please view [#775](https://github.com/frappe/books/issues/775) for more info.
-
diff --git a/fyo/model/types.ts b/fyo/model/types.ts
index 13f0bc406..2868a5c05 100644
--- a/fyo/model/types.ts
+++ b/fyo/model/types.ts
@@ -117,3 +117,13 @@ export type DocStatus =
| 'NotSaved'
| 'Submitted'
| 'Cancelled';
+
+ export type LeadStatus =
+ | ''
+ | 'Open'
+ | 'Replied'
+ | 'Interested'
+ | 'Opportunity'
+ | 'Converted'
+ | 'Quotation'
+ | 'DonotContact'
\ No newline at end of file
diff --git a/models/baseModels/AccountingSettings/AccountingSettings.ts b/models/baseModels/AccountingSettings/AccountingSettings.ts
index 4dc924683..0c7a548c8 100644
--- a/models/baseModels/AccountingSettings/AccountingSettings.ts
+++ b/models/baseModels/AccountingSettings/AccountingSettings.ts
@@ -15,6 +15,7 @@ export class AccountingSettings extends Doc {
enableDiscounting?: boolean;
enableInventory?: boolean;
enablePriceList?: boolean;
+ enableLead?: boolean;
enableFormCustomization?: boolean;
enableInvoiceReturns?: boolean;
@@ -48,6 +49,9 @@ export class AccountingSettings extends Doc {
enableInventory: () => {
return !!this.enableInventory;
},
+ enableLead: () => {
+ return !!this.enableLead;
+ },
enableInvoiceReturns: () => {
return !!this.enableInvoiceReturns;
},
diff --git a/models/baseModels/Invoice/Invoice.ts b/models/baseModels/Invoice/Invoice.ts
index 5e1376b69..e366c00f8 100644
--- a/models/baseModels/Invoice/Invoice.ts
+++ b/models/baseModels/Invoice/Invoice.ts
@@ -165,6 +165,10 @@ export abstract class Invoice extends Transactional {
async afterSubmit() {
await super.afterSubmit();
+ if (this.schemaName === ModelNameEnum.SalesQuote) {
+ return;
+ }
+
// update outstanding amounts
await this.fyo.db.update(this.schemaName, {
name: this.name as string,
diff --git a/models/baseModels/Lead/Lead.ts b/models/baseModels/Lead/Lead.ts
new file mode 100644
index 000000000..3037ab46b
--- /dev/null
+++ b/models/baseModels/Lead/Lead.ts
@@ -0,0 +1,32 @@
+import { Fyo } from 'fyo';
+import { Doc } from 'fyo/model/doc';
+import {
+ Action,
+ LeadStatus,
+ ListViewSettings,
+ ValidationMap,
+} from 'fyo/model/types';
+import { getLeadActions, getLeadStatusColumn } from 'models/helpers';
+import {
+ validateEmail,
+ validatePhoneNumber,
+} from 'fyo/model/validationFunction';
+
+export class Lead extends Doc {
+ status?: LeadStatus;
+
+ validations: ValidationMap = {
+ email: validateEmail,
+ mobile: validatePhoneNumber,
+ };
+
+ static getActions(fyo: Fyo): Action[] {
+ return getLeadActions(fyo);
+ }
+
+ static getListViewSettings(): ListViewSettings {
+ return {
+ columns: ['name', getLeadStatusColumn(), 'email', 'mobile'],
+ };
+ }
+}
diff --git a/models/baseModels/Party/Party.ts b/models/baseModels/Party/Party.ts
index c2a29634f..082ef6a6d 100644
--- a/models/baseModels/Party/Party.ts
+++ b/models/baseModels/Party/Party.ts
@@ -13,9 +13,12 @@ import {
} from 'fyo/model/validationFunction';
import { Money } from 'pesa';
import { PartyRole } from './types';
+import { ModelNameEnum } from 'models/types';
export class Party extends Doc {
role?: PartyRole;
+ party?: string;
+ fromLead?: string;
defaultAccount?: string;
outstandingAmount?: Money;
async updateOutstandingAmount() {
@@ -125,6 +128,25 @@ export class Party extends Doc {
};
}
+ async afterDelete() {
+ await super.afterDelete();
+ if (!this.fromLead) {
+ return;
+ }
+ const leadData = await this.fyo.doc.getDoc(ModelNameEnum.Lead, this.name);
+ await leadData.setAndSync('status', 'Interested');
+ }
+
+ async afterSync() {
+ await super.afterSync();
+ if (!this.fromLead) {
+ return;
+ }
+
+ const leadData = await this.fyo.doc.getDoc(ModelNameEnum.Lead, this.name);
+ await leadData.setAndSync('status', 'Converted');
+ }
+
static getActions(fyo: Fyo): Action[] {
return [
{
diff --git a/models/baseModels/SalesQuote/SalesQuote.ts b/models/baseModels/SalesQuote/SalesQuote.ts
index 2c23731db..e32710143 100644
--- a/models/baseModels/SalesQuote/SalesQuote.ts
+++ b/models/baseModels/SalesQuote/SalesQuote.ts
@@ -1,14 +1,22 @@
import { Fyo } from 'fyo';
import { DocValueMap } from 'fyo/core/types';
-import { Action, ListViewSettings } from 'fyo/model/types';
+import { Action, FiltersMap, ListViewSettings } from 'fyo/model/types';
import { ModelNameEnum } from 'models/types';
import { getQuoteActions, getTransactionStatusColumn } from '../../helpers';
import { Invoice } from '../Invoice/Invoice';
import { SalesQuoteItem } from '../SalesQuoteItem/SalesQuoteItem';
import { Defaults } from '../Defaults/Defaults';
+import { Doc } from 'fyo/model/doc';
+import { Party } from '../Party/Party';
export class SalesQuote extends Invoice {
items?: SalesQuoteItem[];
+ party?: string;
+ name?: string;
+ referenceType?:
+ | ModelNameEnum.SalesInvoice
+ | ModelNameEnum.PurchaseInvoice
+ | ModelNameEnum.Lead;
// This is an inherited method and it must keep the async from the parent
// class
@@ -48,6 +56,20 @@ export class SalesQuote extends Invoice {
return invoice;
}
+ static filters: FiltersMap = {
+ numberSeries: (doc: Doc) => ({ referenceType: doc.schemaName }),
+ };
+
+ async afterSubmit(): Promise
{
+ await super.afterSubmit();
+
+ if (this.referenceType == ModelNameEnum.Lead) {
+ const partyDoc = (await this.loadAndGetLink('party')) as Party;
+
+ await partyDoc.setAndSync('status', 'Quotation');
+ }
+ }
+
static getListViewSettings(): ListViewSettings {
return {
columns: [
diff --git a/models/baseModels/tests/testLead.spec.ts b/models/baseModels/tests/testLead.spec.ts
new file mode 100644
index 000000000..846b63ff5
--- /dev/null
+++ b/models/baseModels/tests/testLead.spec.ts
@@ -0,0 +1,127 @@
+import test from 'tape';
+import { closeTestFyo, getTestFyo, setupTestFyo } from 'tests/helpers';
+import { ModelNameEnum } from 'models/types';
+import { Lead } from '../Lead/Lead';
+import { Party } from '../Party/Party';
+
+const fyo = getTestFyo();
+setupTestFyo(fyo, __filename);
+
+const leadData = {
+ name: 'name2',
+ status: 'Open',
+ email: 'sample@gmail.com',
+ mobile: '1231233545',
+};
+
+const itemData: { name: string; rate: number } = {
+ name: 'Pen',
+ rate: 100,
+};
+
+test('create test docs for Lead', async (t) => {
+ await fyo.doc.getNewDoc(ModelNameEnum.Item, itemData).sync();
+
+ t.ok(
+ fyo.db.exists(ModelNameEnum.Item, itemData.name),
+ `dummy item ${itemData.name} exists`
+ );
+});
+
+test('create a Lead doc', async (t) => {
+ await fyo.doc.getNewDoc(ModelNameEnum.Lead, leadData).sync();
+
+ t.ok(
+ fyo.db.exists(ModelNameEnum.Lead, leadData.name),
+ `${leadData.name} exists`
+ );
+});
+
+test('create Customer from Lead', async (t) => {
+ const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
+
+ const newPartyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
+ ...leadDoc.getValidDict(),
+ fromLead: leadData.name,
+ role: 'Customer',
+ phone: leadData.mobile as string,
+ });
+
+ t.equals(
+ leadDoc.status,
+ 'Open',
+ 'Before Customer created the status must be Open'
+ );
+
+ await newPartyDoc.sync();
+
+ t.equals(
+ leadDoc.status,
+ 'Converted',
+ 'After Customer created the status change to Converted'
+ );
+
+ t.ok(
+ await fyo.db.exists(ModelNameEnum.Party, newPartyDoc.name),
+ 'Customer created from Lead'
+ );
+});
+
+test('create SalesQuote', async (t) => {
+ const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
+
+ const docData = leadDoc.getValidDict(true, true);
+ const newSalesQuoteDoc = fyo.doc.getNewDoc(ModelNameEnum.SalesQuote, {
+ ...docData,
+ party: docData.name,
+ referenceType: ModelNameEnum.Lead,
+ items: [
+ {
+ item: itemData.name,
+ rate: itemData.rate,
+ },
+ ],
+ }) as Lead;
+
+ t.equals(
+ leadDoc.status,
+ 'Converted',
+ 'status must be Open before SQUOT is created'
+ );
+ await newSalesQuoteDoc.sync();
+ await newSalesQuoteDoc.submit();
+
+ t.equals(
+ leadDoc.status,
+ 'Quotation',
+ 'status should change to Quotation after SQUOT submission'
+ );
+
+ t.ok(
+ await fyo.db.exists(ModelNameEnum.SalesQuote, newSalesQuoteDoc.name),
+ 'SalesQuote Created from Lead'
+ );
+});
+
+test('delete Customer then lead status changes to Interested', async (t) => {
+ const partyDoc = (await fyo.doc.getDoc(
+ ModelNameEnum.Party,
+ 'name2'
+ )) as Party;
+ await partyDoc.delete();
+
+ t.equals(
+ await fyo.db.exists(ModelNameEnum.Party, 'name2'),
+ false,
+ 'Customer deleted'
+ );
+ const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
+
+ t.equals(
+ leadDoc.status,
+ 'Interested',
+ 'After Customer deleted the status changed to Interested'
+ );
+});
+
+closeTestFyo(fyo, __filename);
diff --git a/models/helpers.ts b/models/helpers.ts
index c5a25f49d..de8ecb90f 100644
--- a/models/helpers.ts
+++ b/models/helpers.ts
@@ -1,6 +1,12 @@
import { Fyo, t } from 'fyo';
import { Doc } from 'fyo/model/doc';
-import { Action, ColumnConfig, DocStatus, RenderData } from 'fyo/model/types';
+import {
+ Action,
+ ColumnConfig,
+ DocStatus,
+ LeadStatus,
+ RenderData,
+} from 'fyo/model/types';
import { DateTime } from 'luxon';
import { Money } from 'pesa';
import { safeParseFloat } from 'utils/index';
@@ -15,6 +21,7 @@ import { SalesQuote } from './baseModels/SalesQuote/SalesQuote';
import { StockMovement } from './inventory/StockMovement';
import { StockTransfer } from './inventory/StockTransfer';
import { InvoiceStatus, ModelNameEnum } from './types';
+import { Lead } from './baseModels/Lead/Lead';
export function getQuoteActions(
fyo: Fyo,
@@ -23,6 +30,10 @@ export function getQuoteActions(
return [getMakeInvoiceAction(fyo, schemaName)];
}
+export function getLeadActions(fyo: Fyo): Action[] {
+ return [getCreateCustomerAction(fyo), getSalesQuoteAction(fyo)];
+}
+
export function getInvoiceActions(
fyo: Fyo,
schemaName: ModelNameEnum.SalesInvoice | ModelNameEnum.PurchaseInvoice
@@ -108,6 +119,43 @@ export function getMakeInvoiceAction(
};
}
+export function getCreateCustomerAction(fyo: Fyo): Action {
+ return {
+ group: fyo.t`Create`,
+ label: fyo.t`Customer`,
+ action: async (doc: Doc, router) => {
+ const partyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
+ ...doc.getValidDict(),
+ fromLead: doc.name,
+ phone: doc.mobile as string,
+ role: 'Customer',
+ });
+ if (!partyDoc.name) {
+ return;
+ }
+ await router.push(`/edit/Party/${partyDoc.name}`);
+ },
+ };
+}
+
+export function getSalesQuoteAction(fyo: Fyo): Action {
+ return {
+ group: fyo.t`Create`,
+ label: fyo.t`Sales Quote`,
+ action: async (doc, router) => {
+ const data: { party: string | undefined; referenceType: string } = {
+ party: doc.name,
+ referenceType: ModelNameEnum.Lead,
+ };
+ const salesQuoteDoc = fyo.doc.getNewDoc(ModelNameEnum.SalesQuote, data);
+ if (!salesQuoteDoc.name) {
+ return;
+ }
+ await router.push(`/edit/SalesQuote/${salesQuoteDoc.name}`);
+ },
+ };
+}
+
export function getMakePaymentAction(fyo: Fyo): Action {
return {
label: fyo.t`Payment`,
@@ -230,18 +278,42 @@ export function getTransactionStatusColumn(): ColumnConfig {
};
}
+export function getLeadStatusColumn(): ColumnConfig {
+ return {
+ label: t`Status`,
+ fieldname: 'status',
+ fieldtype: 'Select',
+ render(doc) {
+ const status = getLeadStatus(doc) as LeadStatus;
+ const color = statusColor[status] ?? 'gray';
+ const label = getStatusTextOfLead(status);
+
+ return {
+ template: `${label}`,
+ };
+ },
+ };
+}
+
export const statusColor: Record<
- DocStatus | InvoiceStatus,
+ DocStatus | InvoiceStatus | LeadStatus,
string | undefined
> = {
'': 'gray',
Draft: 'gray',
+ Open: 'gray',
+ Replied: 'yellow',
+ Opportunity: 'yellow',
Unpaid: 'orange',
Paid: 'green',
+ Interested: 'yellow',
+ Converted: 'green',
+ Quotation: 'green',
Saved: 'gray',
NotSaved: 'gray',
Submitted: 'green',
Cancelled: 'red',
+ DonotContact: 'red',
Return: 'green',
ReturnIssued: 'green',
};
@@ -271,6 +343,37 @@ export function getStatusText(status: DocStatus | InvoiceStatus): string {
}
}
+export function getStatusTextOfLead(status: LeadStatus): string {
+ switch (status) {
+ case 'Open':
+ return t`Open`;
+ case 'Replied':
+ return t`Replied`;
+ case 'Opportunity':
+ return t`Opportunity`;
+ case 'Interested':
+ return t`Interested`;
+ case 'Converted':
+ return t`Converted`;
+ case 'Quotation':
+ return t`Quotation`;
+ case 'DonotContact':
+ return t`Do not Contact`;
+ default:
+ return '';
+ }
+}
+
+export function getLeadStatus(
+ doc?: Lead | Doc | RenderData
+): LeadStatus | DocStatus {
+ if (!doc) {
+ return '';
+ }
+
+ return doc.status as LeadStatus;
+}
+
export function getDocStatus(
doc?: RenderData | Doc
): DocStatus | InvoiceStatus {
diff --git a/models/index.ts b/models/index.ts
index 77edb6260..a5690a690 100644
--- a/models/index.ts
+++ b/models/index.ts
@@ -9,6 +9,7 @@ import { JournalEntry } from './baseModels/JournalEntry/JournalEntry';
import { JournalEntryAccount } from './baseModels/JournalEntryAccount/JournalEntryAccount';
import { Misc } from './baseModels/Misc';
import { Party } from './baseModels/Party/Party';
+import { Lead } from './baseModels/Lead/Lead';
import { Payment } from './baseModels/Payment/Payment';
import { PaymentFor } from './baseModels/PaymentFor/PaymentFor';
import { PriceList } from './baseModels/PriceList/PriceList';
@@ -53,6 +54,7 @@ export const models = {
JournalEntry,
JournalEntryAccount,
Misc,
+ Lead,
Party,
Payment,
PaymentFor,
diff --git a/models/regionalModels/in/Party.ts b/models/regionalModels/in/Party.ts
index 523cd2a2d..cd560cd5b 100644
--- a/models/regionalModels/in/Party.ts
+++ b/models/regionalModels/in/Party.ts
@@ -4,6 +4,7 @@ import { GSTType } from './types';
export class Party extends BaseParty {
gstin?: string;
+ fromLead?: string;
gstType?: GSTType;
// eslint-disable-next-line @typescript-eslint/require-await
@@ -18,5 +19,6 @@ export class Party extends BaseParty {
hidden: HiddenMap = {
gstin: () => (this.gstType as GSTType) !== 'Registered Regular',
+ fromLead: () => !this.fromLead,
};
}
diff --git a/models/types.ts b/models/types.ts
index 3a83e4bf7..fe591def5 100644
--- a/models/types.ts
+++ b/models/types.ts
@@ -17,6 +17,7 @@ export enum ModelNameEnum {
JournalEntryAccount = 'JournalEntryAccount',
Misc = 'Misc',
NumberSeries = 'NumberSeries',
+ Lead = 'Lead',
Party = 'Party',
Payment = 'Payment',
PaymentFor = 'PaymentFor',
diff --git a/schemas/app/AccountingSettings.json b/schemas/app/AccountingSettings.json
index 469979928..4360352b5 100644
--- a/schemas/app/AccountingSettings.json
+++ b/schemas/app/AccountingSettings.json
@@ -100,6 +100,13 @@
"default": false,
"section": "Features"
},
+ {
+ "fieldname": "enableLead",
+ "label": "Enable Lead",
+ "fieldtype": "Check",
+ "default": false,
+ "section": "Features"
+ },
{
"fieldname": "fiscalYearStart",
"label": "Fiscal Year Start Date",
diff --git a/schemas/app/Lead.json b/schemas/app/Lead.json
new file mode 100644
index 000000000..4dfff2841
--- /dev/null
+++ b/schemas/app/Lead.json
@@ -0,0 +1,76 @@
+{
+ "name": "Lead",
+ "label": "Lead",
+ "naming": "manual",
+ "fields": [
+ {
+ "fieldname": "name",
+ "label": "Name",
+ "fieldtype": "Data",
+ "required": true,
+ "placeholder": "Full Name",
+ "section": "Default"
+ },
+ {
+ "fieldname": "status",
+ "label": "Status",
+ "fieldtype": "Select",
+ "default": "Open",
+ "options": [
+ {
+ "value": "Open",
+ "label": "Open"
+ },
+ {
+ "value": "Replied",
+ "label": "Replied"
+ },
+ {
+ "value": "Interested",
+ "label": "Interested"
+ },
+ {
+ "value": "Opportunity",
+ "label": "Opportunity"
+ },
+ {
+ "value": "Converted",
+ "label": "Converted"
+ },
+ {
+ "value": "Quotation",
+ "label": "Quotation"
+ },
+ {
+ "value": "DonotContact",
+ "label": "Do not Contact"
+ }
+ ],
+ "required": true,
+ "section": "Default"
+ },
+ {
+ "fieldname": "email",
+ "label": "Email",
+ "fieldtype": "Data",
+ "placeholder": "john@doe.com",
+ "section": "Contacts"
+ },
+ {
+ "fieldname": "mobile",
+ "label": "Mobile",
+ "fieldtype": "Data",
+ "placeholder": "Mobile",
+ "section": "Contacts"
+ },
+ {
+ "fieldname": "address",
+ "label": "Address",
+ "fieldtype": "Link",
+ "target": "Address",
+ "create": true,
+ "section": "Contacts"
+ }
+ ],
+ "keywordFields": ["name", "email", "mobile"]
+}
diff --git a/schemas/app/Party.json b/schemas/app/Party.json
index 7f1c4e6e0..f6ccf25b3 100644
--- a/schemas/app/Party.json
+++ b/schemas/app/Party.json
@@ -78,6 +78,14 @@
"create": true,
"section": "Billing"
},
+ {
+ "fieldname": "fromLead",
+ "label": "From Lead",
+ "fieldtype": "Link",
+ "target": "Lead",
+ "readOnly": true,
+ "section": "References"
+ },
{
"fieldname": "taxId",
"label": "Tax ID",
diff --git a/schemas/app/SalesQuote.json b/schemas/app/SalesQuote.json
index 9f5248453..83934eac3 100644
--- a/schemas/app/SalesQuote.json
+++ b/schemas/app/SalesQuote.json
@@ -15,11 +15,29 @@
"default": "SQUOT-",
"section": "Default"
},
+ {
+ "fieldname": "referenceType",
+ "label": "Type",
+ "placeholder": "Type",
+ "fieldtype": "Select",
+ "default": "Party",
+ "options": [
+ {
+ "value": "Party",
+ "label": "Party"
+ },
+ {
+ "value": "Lead",
+ "label": "Lead"
+ }
+ ],
+ "required": true
+ },
{
"fieldname": "party",
"label": "Customer",
- "fieldtype": "Link",
- "target": "Party",
+ "fieldtype": "DynamicLink",
+ "references": "referenceType",
"create": true,
"required": true,
"section": "Default"
diff --git a/schemas/schemas.ts b/schemas/schemas.ts
index 54f4a81ab..a551d529b 100644
--- a/schemas/schemas.ts
+++ b/schemas/schemas.ts
@@ -15,6 +15,7 @@ import JournalEntryAccount from './app/JournalEntryAccount.json';
import Misc from './app/Misc.json';
import NumberSeries from './app/NumberSeries.json';
import Party from './app/Party.json';
+import Lead from './app/Lead.json';
import Payment from './app/Payment.json';
import PaymentFor from './app/PaymentFor.json';
import PriceList from './app/PriceList.json';
@@ -96,6 +97,7 @@ export const appSchemas: Schema[] | SchemaStub[] = [
AccountingLedgerEntry as Schema,
Party as Schema,
+ Lead as Schema,
Address as Schema,
Item as Schema,
UOM as Schema,
diff --git a/src/utils/sidebarConfig.ts b/src/utils/sidebarConfig.ts
index 375575718..dc4aa1592 100644
--- a/src/utils/sidebarConfig.ts
+++ b/src/utils/sidebarConfig.ts
@@ -202,6 +202,13 @@ function getCompleteSidebar(): SidebarConfig {
schemaName: 'Item',
filters: routeFilters.SalesItems,
},
+ {
+ label: t`Lead`,
+ name: 'lead',
+ route: '/list/Lead',
+ schemaName: 'Lead',
+ hidden: () => !fyo.singles.AccountingSettings?.enableLead,
+ },
] as SidebarItem[],
},
{
From 2c784e72d3f802d003062ce6900a8aab2709d0ff Mon Sep 17 00:00:00 2001
From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com>
Date: Fri, 9 Aug 2024 15:56:10 +0530
Subject: [PATCH 2/4] fix: resolved test issues in Lead
---
models/baseModels/Lead/Lead.ts | 19 ++++++++++
models/baseModels/tests/testLead.spec.ts | 48 +++++++++++-------------
models/helpers.ts | 32 ++++++++--------
3 files changed, 57 insertions(+), 42 deletions(-)
diff --git a/models/baseModels/Lead/Lead.ts b/models/baseModels/Lead/Lead.ts
index 3037ab46b..a45cdcb49 100644
--- a/models/baseModels/Lead/Lead.ts
+++ b/models/baseModels/Lead/Lead.ts
@@ -11,6 +11,7 @@ import {
validateEmail,
validatePhoneNumber,
} from 'fyo/model/validationFunction';
+import { ModelNameEnum } from 'models/types';
export class Lead extends Doc {
status?: LeadStatus;
@@ -20,6 +21,24 @@ export class Lead extends Doc {
mobile: validatePhoneNumber,
};
+ createCustomer() {
+ return this.fyo.doc.getNewDoc(ModelNameEnum.Party, {
+ ...this.getValidDict(),
+ fromLead: this.name,
+ phone: this.mobile as string,
+ role: 'Customer',
+ });
+ }
+
+ createSalesQuote() {
+ const data: { party: string | undefined; referenceType: string } = {
+ party: this.name,
+ referenceType: ModelNameEnum.Lead,
+ };
+
+ return this.fyo.doc.getNewDoc(ModelNameEnum.SalesQuote, data);
+ }
+
static getActions(fyo: Fyo): Action[] {
return getLeadActions(fyo);
}
diff --git a/models/baseModels/tests/testLead.spec.ts b/models/baseModels/tests/testLead.spec.ts
index 846b63ff5..e28332c76 100644
--- a/models/baseModels/tests/testLead.spec.ts
+++ b/models/baseModels/tests/testLead.spec.ts
@@ -11,7 +11,7 @@ const leadData = {
name: 'name2',
status: 'Open',
email: 'sample@gmail.com',
- mobile: '1231233545',
+ mobile: '7356203811',
};
const itemData: { name: string; rate: number } = {
@@ -40,29 +40,24 @@ test('create a Lead doc', async (t) => {
test('create Customer from Lead', async (t) => {
const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
- const newPartyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
- ...leadDoc.getValidDict(),
- fromLead: leadData.name,
- role: 'Customer',
- phone: leadData.mobile as string,
- });
+ const newCustomer = leadDoc.createCustomer();
t.equals(
leadDoc.status,
'Open',
- 'Before Customer created the status must be Open'
+ 'status must be Open before Customer is created'
);
- await newPartyDoc.sync();
+ await newCustomer.sync();
t.equals(
leadDoc.status,
'Converted',
- 'After Customer created the status change to Converted'
+ 'status should change to Converted after Customer is created'
);
t.ok(
- await fyo.db.exists(ModelNameEnum.Party, newPartyDoc.name),
+ await fyo.db.exists(ModelNameEnum.Party, newCustomer.name),
'Customer created from Lead'
);
});
@@ -70,26 +65,23 @@ test('create Customer from Lead', async (t) => {
test('create SalesQuote', async (t) => {
const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
- const docData = leadDoc.getValidDict(true, true);
- const newSalesQuoteDoc = fyo.doc.getNewDoc(ModelNameEnum.SalesQuote, {
- ...docData,
- party: docData.name,
- referenceType: ModelNameEnum.Lead,
- items: [
- {
- item: itemData.name,
- rate: itemData.rate,
- },
- ],
- }) as Lead;
+ const newSalesQuote = leadDoc.createSalesQuote();
+
+ newSalesQuote.items = [];
+ newSalesQuote.append('items', {
+ item: itemData.name,
+ quantity: 1,
+ rate: itemData.rate,
+ });
t.equals(
leadDoc.status,
'Converted',
'status must be Open before SQUOT is created'
);
- await newSalesQuoteDoc.sync();
- await newSalesQuoteDoc.submit();
+
+ await newSalesQuote.sync();
+ await newSalesQuote.submit();
t.equals(
leadDoc.status,
@@ -98,7 +90,7 @@ test('create SalesQuote', async (t) => {
);
t.ok(
- await fyo.db.exists(ModelNameEnum.SalesQuote, newSalesQuoteDoc.name),
+ await fyo.db.exists(ModelNameEnum.SalesQuote, newSalesQuote.name),
'SalesQuote Created from Lead'
);
});
@@ -108,6 +100,7 @@ test('delete Customer then lead status changes to Interested', async (t) => {
ModelNameEnum.Party,
'name2'
)) as Party;
+
await partyDoc.delete();
t.equals(
@@ -115,12 +108,13 @@ test('delete Customer then lead status changes to Interested', async (t) => {
false,
'Customer deleted'
);
+
const leadDoc = (await fyo.doc.getDoc(ModelNameEnum.Lead, 'name2')) as Lead;
t.equals(
leadDoc.status,
'Interested',
- 'After Customer deleted the status changed to Interested'
+ 'status should change to Interested after Customer is deleted'
);
});
diff --git a/models/helpers.ts b/models/helpers.ts
index de8ecb90f..ad46ca3d3 100644
--- a/models/helpers.ts
+++ b/models/helpers.ts
@@ -124,16 +124,12 @@ export function getCreateCustomerAction(fyo: Fyo): Action {
group: fyo.t`Create`,
label: fyo.t`Customer`,
action: async (doc: Doc, router) => {
- const partyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
- ...doc.getValidDict(),
- fromLead: doc.name,
- phone: doc.mobile as string,
- role: 'Customer',
- });
- if (!partyDoc.name) {
+ const customerData = (doc as Lead).createCustomer();
+
+ if (!customerData.name) {
return;
}
- await router.push(`/edit/Party/${partyDoc.name}`);
+ await router.push(`/edit/Party/${customerData.name}`);
},
};
}
@@ -143,15 +139,11 @@ export function getSalesQuoteAction(fyo: Fyo): Action {
group: fyo.t`Create`,
label: fyo.t`Sales Quote`,
action: async (doc, router) => {
- const data: { party: string | undefined; referenceType: string } = {
- party: doc.name,
- referenceType: ModelNameEnum.Lead,
- };
- const salesQuoteDoc = fyo.doc.getNewDoc(ModelNameEnum.SalesQuote, data);
- if (!salesQuoteDoc.name) {
+ const salesQuoteData = (doc as Lead).createSalesQuote();
+ if (!salesQuoteData.name) {
return;
}
- await router.push(`/edit/SalesQuote/${salesQuoteDoc.name}`);
+ await router.push(`/edit/SalesQuote/${salesQuoteData.name}`);
},
};
}
@@ -295,6 +287,16 @@ export function getLeadStatusColumn(): ColumnConfig {
};
}
+// export async function createCustomer(fyo: Fyo, doc: Doc) {
+// const partyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
+// ...doc.getValidDict(),
+// fromLead: doc.name,
+// phone: doc.mobile as string,
+// role: 'Customer',
+// });
+// return partyDoc;
+// }
+
export const statusColor: Record<
DocStatus | InvoiceStatus | LeadStatus,
string | undefined
From e2d6c2a04862ca65f027e2a3df2d024bbc3e56ed Mon Sep 17 00:00:00 2001
From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com>
Date: Mon, 12 Aug 2024 10:50:05 +0530
Subject: [PATCH 3/4] fix: resolved test issues in Lead
---
models/baseModels/tests/testLead.spec.ts | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/models/baseModels/tests/testLead.spec.ts b/models/baseModels/tests/testLead.spec.ts
index e28332c76..abc575cd4 100644
--- a/models/baseModels/tests/testLead.spec.ts
+++ b/models/baseModels/tests/testLead.spec.ts
@@ -11,7 +11,7 @@ const leadData = {
name: 'name2',
status: 'Open',
email: 'sample@gmail.com',
- mobile: '7356203811',
+ mobile: '1234567890',
};
const itemData: { name: string; rate: number } = {
From 3e10ba6d306e83b54b773e764c59ee8d554107aa Mon Sep 17 00:00:00 2001
From: AbleKSaju <126228406+AbleKSaju@users.noreply.github.com>
Date: Tue, 13 Aug 2024 09:31:54 +0530
Subject: [PATCH 4/4] fix: removed unusual comments
---
models/helpers.ts | 10 ----------
1 file changed, 10 deletions(-)
diff --git a/models/helpers.ts b/models/helpers.ts
index ad46ca3d3..2c67256c5 100644
--- a/models/helpers.ts
+++ b/models/helpers.ts
@@ -287,16 +287,6 @@ export function getLeadStatusColumn(): ColumnConfig {
};
}
-// export async function createCustomer(fyo: Fyo, doc: Doc) {
-// const partyDoc = fyo.doc.getNewDoc(ModelNameEnum.Party, {
-// ...doc.getValidDict(),
-// fromLead: doc.name,
-// phone: doc.mobile as string,
-// role: 'Customer',
-// });
-// return partyDoc;
-// }
-
export const statusColor: Record<
DocStatus | InvoiceStatus | LeadStatus,
string | undefined