Merge branch 'develop' into fix_allocated_amt_in_PE
This commit is contained in:
commit
fee9a56d4e
@ -592,6 +592,7 @@ class JournalEntry(AccountsController):
|
|||||||
|
|
||||||
self.validate_total_debit_and_credit()
|
self.validate_total_debit_and_credit()
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def get_outstanding_invoices(self):
|
def get_outstanding_invoices(self):
|
||||||
self.set('accounts', [])
|
self.set('accounts', [])
|
||||||
total = 0
|
total = 0
|
||||||
|
@ -17,10 +17,12 @@ class AmazonMWSSettings(Document):
|
|||||||
else:
|
else:
|
||||||
self.enable_sync = 0
|
self.enable_sync = 0
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def get_products_details(self):
|
def get_products_details(self):
|
||||||
if self.enable_amazon == 1:
|
if self.enable_amazon == 1:
|
||||||
frappe.enqueue('erpnext.erpnext_integrations.doctype.amazon_mws_settings.amazon_methods.get_products_details')
|
frappe.enqueue('erpnext.erpnext_integrations.doctype.amazon_mws_settings.amazon_methods.get_products_details')
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def get_order_details(self):
|
def get_order_details(self):
|
||||||
if self.enable_amazon == 1:
|
if self.enable_amazon == 1:
|
||||||
after_date = dateutil.parser.parse(self.after_date).strftime("%Y-%m-%d")
|
after_date = dateutil.parser.parse(self.after_date).strftime("%Y-%m-%d")
|
||||||
@ -40,4 +42,4 @@ def setup_custom_fields():
|
|||||||
fieldtype='Data', insert_after='title', read_only=1, print_hide=1)]
|
fieldtype='Data', insert_after='title', read_only=1, print_hide=1)]
|
||||||
}
|
}
|
||||||
|
|
||||||
create_custom_fields(custom_fields)
|
create_custom_fields(custom_fields)
|
||||||
|
@ -50,6 +50,7 @@ class TherapyType(Document):
|
|||||||
|
|
||||||
self.db_set('change_in_item', 0)
|
self.db_set('change_in_item', 0)
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def add_exercises(self):
|
def add_exercises(self):
|
||||||
exercises = self.get_exercises_for_body_parts()
|
exercises = self.get_exercises_for_body_parts()
|
||||||
last_idx = max([cint(d.idx) for d in self.get('exercises')] or [0,])
|
last_idx = max([cint(d.idx) for d in self.get('exercises')] or [0,])
|
||||||
|
@ -62,6 +62,7 @@ class HolidayList(Document):
|
|||||||
|
|
||||||
return date_list
|
return date_list
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def clear_table(self):
|
def clear_table(self):
|
||||||
self.set('holidays', [])
|
self.set('holidays', [])
|
||||||
|
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
"retirement_age",
|
"retirement_age",
|
||||||
"emp_created_by",
|
"emp_created_by",
|
||||||
"column_break_4",
|
"column_break_4",
|
||||||
|
"standard_working_hours",
|
||||||
"stop_birthday_reminders",
|
"stop_birthday_reminders",
|
||||||
"expense_approver_mandatory_in_expense_claim",
|
"expense_approver_mandatory_in_expense_claim",
|
||||||
"leave_settings",
|
"leave_settings",
|
||||||
@ -143,13 +144,19 @@
|
|||||||
"fieldname": "send_leave_notification",
|
"fieldname": "send_leave_notification",
|
||||||
"fieldtype": "Check",
|
"fieldtype": "Check",
|
||||||
"label": "Send Leave Notification"
|
"label": "Send Leave Notification"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "8",
|
||||||
|
"fieldname": "standard_working_hours",
|
||||||
|
"fieldtype": "Int",
|
||||||
|
"label": "Standard Working Hours"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"icon": "fa fa-cog",
|
"icon": "fa fa-cog",
|
||||||
"idx": 1,
|
"idx": 1,
|
||||||
"issingle": 1,
|
"issingle": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-03-14 02:04:22.907159",
|
"modified": "2021-04-16 15:45:18.467699",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "HR",
|
"module": "HR",
|
||||||
"name": "HR Settings",
|
"name": "HR Settings",
|
||||||
|
@ -29,6 +29,7 @@ class LeaveControlPanel(Document):
|
|||||||
frappe.throw(_("{0} is required").format(self.meta.get_label(f)))
|
frappe.throw(_("{0} is required").format(self.meta.get_label(f)))
|
||||||
self.validate_from_to_dates('from_date', 'to_date')
|
self.validate_from_to_dates('from_date', 'to_date')
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def allocate_leave(self):
|
def allocate_leave(self):
|
||||||
self.validate_values()
|
self.validate_values()
|
||||||
leave_allocated_for = []
|
leave_allocated_for = []
|
||||||
|
@ -32,7 +32,8 @@ frappe.ui.form.on("Task", {
|
|||||||
|
|
||||||
frm.set_query("parent_task", function () {
|
frm.set_query("parent_task", function () {
|
||||||
let filters = {
|
let filters = {
|
||||||
"is_group": 1
|
"is_group": 1,
|
||||||
|
"name": ["!=", frm.doc.name]
|
||||||
};
|
};
|
||||||
if (frm.doc.project) filters["project"] = frm.doc.project;
|
if (frm.doc.project) filters["project"] = frm.doc.project;
|
||||||
return {
|
return {
|
||||||
|
@ -151,11 +151,11 @@ class TestTimesheet(unittest.TestCase):
|
|||||||
settings.save()
|
settings.save()
|
||||||
|
|
||||||
|
|
||||||
def make_salary_structure_for_timesheet(employee):
|
def make_salary_structure_for_timesheet(employee, company=None):
|
||||||
salary_structure_name = "Timesheet Salary Structure Test"
|
salary_structure_name = "Timesheet Salary Structure Test"
|
||||||
frequency = "Monthly"
|
frequency = "Monthly"
|
||||||
|
|
||||||
salary_structure = make_salary_structure(salary_structure_name, frequency, dont_submit=True)
|
salary_structure = make_salary_structure(salary_structure_name, frequency, company=company, dont_submit=True)
|
||||||
salary_structure.salary_component = "Timesheet Component"
|
salary_structure.salary_component = "Timesheet Component"
|
||||||
salary_structure.salary_slip_based_on_timesheet = 1
|
salary_structure.salary_slip_based_on_timesheet = 1
|
||||||
salary_structure.hour_rate = 50.0
|
salary_structure.hour_rate = 50.0
|
||||||
|
@ -0,0 +1,48 @@
|
|||||||
|
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
/* eslint-disable */
|
||||||
|
|
||||||
|
frappe.query_reports["Project Profitability"] = {
|
||||||
|
"filters": [
|
||||||
|
{
|
||||||
|
"fieldname": "company",
|
||||||
|
"label": __("Company"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Company",
|
||||||
|
"default": frappe.defaults.get_user_default("Company"),
|
||||||
|
"reqd": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "start_date",
|
||||||
|
"label": __("Start Date"),
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"reqd": 1,
|
||||||
|
"default": frappe.datetime.add_months(frappe.datetime.get_today(), -1)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "end_date",
|
||||||
|
"label": __("End Date"),
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"reqd": 1,
|
||||||
|
"default": frappe.datetime.now_date()
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "customer_name",
|
||||||
|
"label": __("Customer"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Customer"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "employee",
|
||||||
|
"label": __("Employee"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Employee"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "project",
|
||||||
|
"label": __("Project"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Project"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
@ -0,0 +1,44 @@
|
|||||||
|
{
|
||||||
|
"add_total_row": 0,
|
||||||
|
"columns": [],
|
||||||
|
"creation": "2021-04-16 15:50:28.914872",
|
||||||
|
"disable_prepared_report": 0,
|
||||||
|
"disabled": 0,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Report",
|
||||||
|
"filters": [],
|
||||||
|
"idx": 0,
|
||||||
|
"is_standard": "Yes",
|
||||||
|
"modified": "2021-04-16 15:50:48.490866",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "Projects",
|
||||||
|
"name": "Project Profitability",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"prepared_report": 0,
|
||||||
|
"ref_doctype": "Timesheet",
|
||||||
|
"report_name": "Project Profitability",
|
||||||
|
"report_type": "Script Report",
|
||||||
|
"roles": [
|
||||||
|
{
|
||||||
|
"role": "HR User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "Accounts User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "Employee"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "Projects User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "Manufacturing User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "Employee Self Service"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"role": "HR Manager"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -0,0 +1,210 @@
|
|||||||
|
# Copyright (c) 2013, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
import frappe
|
||||||
|
from frappe import _
|
||||||
|
|
||||||
|
def execute(filters=None):
|
||||||
|
columns, data = [], []
|
||||||
|
data = get_data(filters)
|
||||||
|
columns = get_columns()
|
||||||
|
charts = get_chart_data(data)
|
||||||
|
return columns, data, None, charts
|
||||||
|
|
||||||
|
def get_data(filters):
|
||||||
|
data = get_rows(filters)
|
||||||
|
data = calculate_cost_and_profit(data)
|
||||||
|
return data
|
||||||
|
|
||||||
|
def get_rows(filters):
|
||||||
|
conditions = get_conditions(filters)
|
||||||
|
standard_working_hours = frappe.db.get_single_value("HR Settings", "standard_working_hours")
|
||||||
|
if not standard_working_hours:
|
||||||
|
msg = _("The metrics for this report are calculated based on the Standard Working Hours. Please set {0} in {1}.").format(
|
||||||
|
frappe.bold("Standard Working Hours"), frappe.utils.get_link_to_form("HR Settings", "HR Settings"))
|
||||||
|
|
||||||
|
frappe.msgprint(msg)
|
||||||
|
return []
|
||||||
|
|
||||||
|
sql = """
|
||||||
|
SELECT
|
||||||
|
*
|
||||||
|
FROM
|
||||||
|
(SELECT
|
||||||
|
si.customer_name,si.base_grand_total,
|
||||||
|
si.name as voucher_no,tabTimesheet.employee,
|
||||||
|
tabTimesheet.title as employee_name,tabTimesheet.parent_project as project,
|
||||||
|
tabTimesheet.start_date,tabTimesheet.end_date,
|
||||||
|
tabTimesheet.total_billed_hours,tabTimesheet.name as timesheet,
|
||||||
|
ss.base_gross_pay,ss.total_working_days,
|
||||||
|
tabTimesheet.total_billed_hours/(ss.total_working_days * {0}) as utilization
|
||||||
|
FROM
|
||||||
|
`tabSalary Slip Timesheet` as sst join `tabTimesheet` on tabTimesheet.name = sst.time_sheet
|
||||||
|
join `tabSales Invoice Timesheet` as sit on sit.time_sheet = tabTimesheet.name
|
||||||
|
join `tabSales Invoice` as si on si.name = sit.parent and si.status != "Cancelled"
|
||||||
|
join `tabSalary Slip` as ss on ss.name = sst.parent and ss.status != "Cancelled" """.format(standard_working_hours)
|
||||||
|
if conditions:
|
||||||
|
sql += """
|
||||||
|
WHERE
|
||||||
|
{0}) as t""".format(conditions)
|
||||||
|
return frappe.db.sql(sql,filters, as_dict=True)
|
||||||
|
|
||||||
|
def calculate_cost_and_profit(data):
|
||||||
|
for row in data:
|
||||||
|
row.fractional_cost = row.base_gross_pay * row.utilization
|
||||||
|
row.profit = row.base_grand_total - row.base_gross_pay * row.utilization
|
||||||
|
return data
|
||||||
|
|
||||||
|
def get_conditions(filters):
|
||||||
|
conditions = []
|
||||||
|
|
||||||
|
if filters.get("company"):
|
||||||
|
conditions.append("tabTimesheet.company={0}".format(frappe.db.escape(filters.get("company"))))
|
||||||
|
|
||||||
|
if filters.get("start_date"):
|
||||||
|
conditions.append("tabTimesheet.start_date>='{0}'".format(filters.get("start_date")))
|
||||||
|
|
||||||
|
if filters.get("end_date"):
|
||||||
|
conditions.append("tabTimesheet.end_date<='{0}'".format(filters.get("end_date")))
|
||||||
|
|
||||||
|
if filters.get("customer_name"):
|
||||||
|
conditions.append("si.customer_name={0}".format(frappe.db.escape(filters.get("customer_name"))))
|
||||||
|
|
||||||
|
if filters.get("employee"):
|
||||||
|
conditions.append("tabTimesheet.employee={0}".format(frappe.db.escape(filters.get("employee"))))
|
||||||
|
|
||||||
|
if filters.get("project"):
|
||||||
|
conditions.append("tabTimesheet.parent_project={0}".format(frappe.db.escape(filters.get("project"))))
|
||||||
|
|
||||||
|
conditions = " and ".join(conditions)
|
||||||
|
return conditions
|
||||||
|
|
||||||
|
def get_chart_data(data):
|
||||||
|
if not data:
|
||||||
|
return None
|
||||||
|
|
||||||
|
labels = []
|
||||||
|
utilization = []
|
||||||
|
|
||||||
|
for entry in data:
|
||||||
|
labels.append(entry.get("employee_name") + " - " + str(entry.get("end_date")))
|
||||||
|
utilization.append(entry.get("utilization"))
|
||||||
|
|
||||||
|
charts = {
|
||||||
|
"data": {
|
||||||
|
"labels": labels,
|
||||||
|
"datasets": [
|
||||||
|
{
|
||||||
|
"name": "Utilization",
|
||||||
|
"values": utilization
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"type": "bar",
|
||||||
|
"colors": ["#84BDD5"]
|
||||||
|
}
|
||||||
|
return charts
|
||||||
|
|
||||||
|
def get_columns():
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
"fieldname": "customer_name",
|
||||||
|
"label": _("Customer"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Customer",
|
||||||
|
"width": 150
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "employee",
|
||||||
|
"label": _("Employee"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Employee",
|
||||||
|
"width": 130
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "employee_name",
|
||||||
|
"label": _("Employee Name"),
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"width": 120
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "voucher_no",
|
||||||
|
"label": _("Sales Invoice"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Sales Invoice",
|
||||||
|
"width": 120
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "timesheet",
|
||||||
|
"label": _("Timesheet"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Timesheet",
|
||||||
|
"width": 120
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "project",
|
||||||
|
"label": _("Project"),
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Project",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "base_grand_total",
|
||||||
|
"label": _("Bill Amount"),
|
||||||
|
"fieldtype": "Currency",
|
||||||
|
"options": "currency",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "base_gross_pay",
|
||||||
|
"label": _("Cost"),
|
||||||
|
"fieldtype": "Currency",
|
||||||
|
"options": "currency",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "profit",
|
||||||
|
"label": _("Profit"),
|
||||||
|
"fieldtype": "Currency",
|
||||||
|
"options": "currency",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "utilization",
|
||||||
|
"label": _("Utilization"),
|
||||||
|
"fieldtype": "Percentage",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "fractional_cost",
|
||||||
|
"label": _("Fractional Cost"),
|
||||||
|
"fieldtype": "Int",
|
||||||
|
"width": 120
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "total_billed_hours",
|
||||||
|
"label": _("Total Billed Hours"),
|
||||||
|
"fieldtype": "Int",
|
||||||
|
"width": 150
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "start_date",
|
||||||
|
"label": _("Start Date"),
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "end_date",
|
||||||
|
"label": _("End Date"),
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"width": 100
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": _("Currency"),
|
||||||
|
"fieldname": "currency",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"options": "Currency",
|
||||||
|
"width": 80
|
||||||
|
}
|
||||||
|
]
|
@ -0,0 +1,56 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
import unittest
|
||||||
|
import frappe
|
||||||
|
from frappe.utils import getdate, nowdate
|
||||||
|
from erpnext.hr.doctype.employee.test_employee import make_employee
|
||||||
|
from erpnext.projects.doctype.timesheet.test_timesheet import make_salary_structure_for_timesheet, make_timesheet
|
||||||
|
from erpnext.projects.doctype.timesheet.timesheet import make_salary_slip, make_sales_invoice
|
||||||
|
from erpnext.projects.report.project_profitability.project_profitability import execute
|
||||||
|
|
||||||
|
class TestProjectProfitability(unittest.TestCase):
|
||||||
|
@classmethod
|
||||||
|
def setUp(self):
|
||||||
|
emp = make_employee('test_employee_9@salary.com', company='_Test Company')
|
||||||
|
if not frappe.db.exists('Salary Component', 'Timesheet Component'):
|
||||||
|
frappe.get_doc({'doctype': 'Salary Component', 'salary_component': 'Timesheet Component'}).insert()
|
||||||
|
make_salary_structure_for_timesheet(emp, company='_Test Company')
|
||||||
|
self.timesheet = make_timesheet(emp, simulate = True, billable=1)
|
||||||
|
self.salary_slip = make_salary_slip(self.timesheet.name)
|
||||||
|
self.salary_slip.submit()
|
||||||
|
self.sales_invoice = make_sales_invoice(self.timesheet.name, '_Test Item', '_Test Customer')
|
||||||
|
self.sales_invoice.due_date = nowdate()
|
||||||
|
self.sales_invoice.submit()
|
||||||
|
|
||||||
|
def test_project_profitability(self):
|
||||||
|
filters = {
|
||||||
|
'company': '_Test Company',
|
||||||
|
'start_date': getdate(),
|
||||||
|
'end_date': getdate()
|
||||||
|
}
|
||||||
|
|
||||||
|
report = execute(filters)
|
||||||
|
|
||||||
|
row = report[1][0]
|
||||||
|
timesheet = frappe.get_doc("Timesheet", self.timesheet.name)
|
||||||
|
|
||||||
|
self.assertEqual(self.sales_invoice.customer, row.customer_name)
|
||||||
|
self.assertEqual(timesheet.title, row.employee_name)
|
||||||
|
self.assertEqual(self.sales_invoice.base_grand_total, row.base_grand_total)
|
||||||
|
self.assertEqual(self.salary_slip.base_gross_pay, row.base_gross_pay)
|
||||||
|
self.assertEqual(timesheet.total_billed_hours, row.total_billed_hours)
|
||||||
|
self.assertEqual(self.salary_slip.total_working_days, row.total_working_days)
|
||||||
|
|
||||||
|
standard_working_hours = frappe.db.get_single_value("HR Settings", "standard_working_hours")
|
||||||
|
utilization = timesheet.total_billed_hours/(self.salary_slip.total_working_days * standard_working_hours)
|
||||||
|
self.assertEqual(utilization, row.utilization)
|
||||||
|
|
||||||
|
profit = self.sales_invoice.base_grand_total - self.salary_slip.base_gross_pay * utilization
|
||||||
|
self.assertEqual(profit, row.profit)
|
||||||
|
|
||||||
|
fractional_cost = self.salary_slip.base_gross_pay * utilization
|
||||||
|
self.assertEqual(fractional_cost, row.fractional_cost)
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
frappe.get_doc("Sales Invoice", self.sales_invoice.name).cancel()
|
||||||
|
frappe.get_doc("Salary Slip", self.salary_slip.name).cancel()
|
||||||
|
frappe.get_doc("Timesheet", self.timesheet.name).cancel()
|
@ -130,6 +130,16 @@
|
|||||||
"onboard": 1,
|
"onboard": 1,
|
||||||
"type": "Link"
|
"type": "Link"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"dependencies": "Timesheet, Sales Invoice, Salary Slip",
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 1,
|
||||||
|
"label": "Project Profitability",
|
||||||
|
"link_to": "Project Profitability",
|
||||||
|
"link_type": "Report",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"dependencies": "Project",
|
"dependencies": "Project",
|
||||||
"hidden": 0,
|
"hidden": 0,
|
||||||
@ -161,7 +171,7 @@
|
|||||||
"type": "Link"
|
"type": "Link"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"modified": "2021-03-26 16:32:00.628561",
|
"modified": "2021-04-16 16:27:16.548780",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Projects",
|
"module": "Projects",
|
||||||
"name": "Projects",
|
"name": "Projects",
|
||||||
|
@ -124,6 +124,7 @@ class ImportSupplierInvoice(Document):
|
|||||||
if disc_line.find("Percentuale"):
|
if disc_line.find("Percentuale"):
|
||||||
invoices_args["total_discount"] += flt((flt(disc_line.Percentuale.text) / 100) * (rate * qty))
|
invoices_args["total_discount"] += flt((flt(disc_line.Percentuale.text) / 100) * (rate * qty))
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def process_file_data(self):
|
def process_file_data(self):
|
||||||
self.status = "Processing File Data"
|
self.status = "Processing File Data"
|
||||||
self.save()
|
self.save()
|
||||||
@ -400,4 +401,4 @@ def get_full_path(file_name):
|
|||||||
elif not self.file_url:
|
elif not self.file_url:
|
||||||
frappe.throw(_("There is some problem with the file url: {0}").format(file_path))
|
frappe.throw(_("There is some problem with the file url: {0}").format(file_path))
|
||||||
|
|
||||||
return file_path
|
return file_path
|
||||||
|
@ -12,6 +12,7 @@ from frappe.model.document import Document
|
|||||||
from frappe.core.doctype.sms_settings.sms_settings import send_sms
|
from frappe.core.doctype.sms_settings.sms_settings import send_sms
|
||||||
|
|
||||||
class SMSCenter(Document):
|
class SMSCenter(Document):
|
||||||
|
@frappe.whitelist()
|
||||||
def create_receiver_list(self):
|
def create_receiver_list(self):
|
||||||
rec, where_clause = '', ''
|
rec, where_clause = '', ''
|
||||||
if self.send_to == 'All Customer Contact':
|
if self.send_to == 'All Customer Contact':
|
||||||
@ -73,6 +74,7 @@ class SMSCenter(Document):
|
|||||||
|
|
||||||
return receiver_nos
|
return receiver_nos
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def send_sms(self):
|
def send_sms(self):
|
||||||
receiver_list = []
|
receiver_list = []
|
||||||
if not self.message:
|
if not self.message:
|
||||||
|
@ -159,6 +159,7 @@ class NamingSeries(Document):
|
|||||||
if frappe.db.get_value('Series', series, 'name', order_by="name") == None:
|
if frappe.db.get_value('Series', series, 'name', order_by="name") == None:
|
||||||
frappe.db.sql("insert into tabSeries (name, current) values (%s, 0)", (series))
|
frappe.db.sql("insert into tabSeries (name, current) values (%s, 0)", (series))
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def update_series_start(self):
|
def update_series_start(self):
|
||||||
if self.prefix:
|
if self.prefix:
|
||||||
prefix = self.parse_naming_series()
|
prefix = self.parse_naming_series()
|
||||||
|
@ -221,6 +221,7 @@ class PurchaseReceipt(BuyingController):
|
|||||||
self.ignore_linked_doctypes = ('GL Entry', 'Stock Ledger Entry', 'Repost Item Valuation')
|
self.ignore_linked_doctypes = ('GL Entry', 'Stock Ledger Entry', 'Repost Item Valuation')
|
||||||
self.delete_auto_created_batches()
|
self.delete_auto_created_batches()
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def get_current_stock(self):
|
def get_current_stock(self):
|
||||||
for d in self.get('supplied_items'):
|
for d in self.get('supplied_items'):
|
||||||
if self.supplier_warehouse:
|
if self.supplier_warehouse:
|
||||||
|
@ -398,8 +398,12 @@ class StockEntry(StockController):
|
|||||||
and item_code = %s
|
and item_code = %s
|
||||||
and ifnull(s_warehouse,'')='' """ % (", ".join(["%s" * len(other_ste)]), "%s"), args)[0][0]
|
and ifnull(s_warehouse,'')='' """ % (", ".join(["%s" * len(other_ste)]), "%s"), args)[0][0]
|
||||||
if fg_qty_already_entered and fg_qty_already_entered >= qty:
|
if fg_qty_already_entered and fg_qty_already_entered >= qty:
|
||||||
frappe.throw(_("Stock Entries already created for Work Order ")
|
frappe.throw(
|
||||||
+ self.work_order + ":" + ", ".join(other_ste), DuplicateEntryForWorkOrderError)
|
_("Stock Entries already created for Work Order {0}: {1}").format(
|
||||||
|
self.work_order, ", ".join(other_ste)
|
||||||
|
),
|
||||||
|
DuplicateEntryForWorkOrderError,
|
||||||
|
)
|
||||||
|
|
||||||
def set_actual_qty(self):
|
def set_actual_qty(self):
|
||||||
allow_negative_stock = cint(frappe.db.get_value("Stock Settings", None, "allow_negative_stock"))
|
allow_negative_stock = cint(frappe.db.get_value("Stock Settings", None, "allow_negative_stock"))
|
||||||
@ -435,6 +439,7 @@ class StockEntry(StockController):
|
|||||||
if transferred_serial_no:
|
if transferred_serial_no:
|
||||||
d.serial_no = transferred_serial_no
|
d.serial_no = transferred_serial_no
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
def get_stock_and_rate(self):
|
def get_stock_and_rate(self):
|
||||||
"""
|
"""
|
||||||
Updates rate and availability of all the items.
|
Updates rate and availability of all the items.
|
||||||
|
Loading…
Reference in New Issue
Block a user