2019-03-11 17:43:44 +05:30
|
|
|
# Copyright (c) 2013, Frappe Technologies Pvt. Ltd. and contributors
|
|
|
|
# For license information, please see license.txt
|
|
|
|
|
|
|
|
|
|
|
|
import frappe
|
|
|
|
from frappe import _
|
2021-09-02 16:44:59 +05:30
|
|
|
from frappe.utils import flt, time_diff_in_hours
|
|
|
|
|
2019-03-11 17:43:44 +05:30
|
|
|
|
|
|
|
def get_columns():
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
"label": _("Employee ID"),
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"fieldname": "employee",
|
|
|
|
"options": "Employee",
|
2022-03-28 18:52:46 +05:30
|
|
|
"width": 300,
|
2019-03-11 17:43:44 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Employee Name"),
|
|
|
|
"fieldtype": "data",
|
|
|
|
"fieldname": "employee_name",
|
|
|
|
"hidden": 1,
|
2022-03-28 18:52:46 +05:30
|
|
|
"width": 200,
|
2019-03-11 17:43:44 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
"label": _("Timesheet"),
|
|
|
|
"fieldtype": "Link",
|
|
|
|
"fieldname": "timesheet",
|
|
|
|
"options": "Timesheet",
|
2022-03-28 18:52:46 +05:30
|
|
|
"width": 150,
|
2019-03-11 17:43:44 +05:30
|
|
|
},
|
2022-03-28 18:52:46 +05:30
|
|
|
{"label": _("Working Hours"), "fieldtype": "Float", "fieldname": "total_hours", "width": 150},
|
2019-03-11 17:43:44 +05:30
|
|
|
{
|
2019-06-13 16:08:01 +05:30
|
|
|
"label": _("Billable Hours"),
|
2019-04-09 11:31:11 +05:30
|
|
|
"fieldtype": "Float",
|
2019-06-13 16:08:01 +05:30
|
|
|
"fieldname": "total_billable_hours",
|
2022-03-28 18:52:46 +05:30
|
|
|
"width": 150,
|
2019-03-11 17:43:44 +05:30
|
|
|
},
|
2022-03-28 18:52:46 +05:30
|
|
|
{"label": _("Billing Amount"), "fieldtype": "Currency", "fieldname": "amount", "width": 150},
|
2019-03-11 17:43:44 +05:30
|
|
|
]
|
|
|
|
|
2022-03-28 18:52:46 +05:30
|
|
|
|
2019-03-11 17:43:44 +05:30
|
|
|
def get_data(filters):
|
|
|
|
data = []
|
2022-03-28 18:52:46 +05:30
|
|
|
if filters.from_date > filters.to_date:
|
2020-04-08 09:32:41 +05:30
|
|
|
frappe.msgprint(_("From Date can not be greater than To Date"))
|
2019-05-02 18:11:08 +05:30
|
|
|
return data
|
2019-03-11 17:43:44 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
timesheets = get_timesheets(filters)
|
|
|
|
|
|
|
|
filters.from_date = frappe.utils.get_datetime(filters.from_date)
|
2022-03-28 18:52:46 +05:30
|
|
|
filters.to_date = frappe.utils.add_to_date(
|
|
|
|
frappe.utils.get_datetime(filters.to_date), days=1, seconds=-1
|
|
|
|
)
|
2019-05-02 18:11:08 +05:30
|
|
|
|
|
|
|
timesheet_details = get_timesheet_details(filters, timesheets.keys())
|
|
|
|
|
|
|
|
for ts, ts_details in timesheet_details.items():
|
2019-03-11 17:43:44 +05:30
|
|
|
total_hours = 0
|
2019-05-02 18:11:08 +05:30
|
|
|
total_billing_hours = 0
|
2019-03-11 17:43:44 +05:30
|
|
|
total_amount = 0
|
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
for row in ts_details:
|
|
|
|
from_time, to_time = filters.from_date, filters.to_date
|
|
|
|
|
|
|
|
if row.to_time < from_time or row.from_time > to_time:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if row.from_time > from_time:
|
|
|
|
from_time = row.from_time
|
|
|
|
|
|
|
|
if row.to_time < to_time:
|
|
|
|
to_time = row.to_time
|
|
|
|
|
|
|
|
activity_duration, billing_duration = get_billable_and_total_duration(row, from_time, to_time)
|
|
|
|
|
|
|
|
total_hours += activity_duration
|
|
|
|
total_billing_hours += billing_duration
|
|
|
|
total_amount += billing_duration * flt(row.billing_rate)
|
|
|
|
|
|
|
|
if total_hours:
|
2022-03-28 18:52:46 +05:30
|
|
|
data.append(
|
|
|
|
{
|
|
|
|
"employee": timesheets.get(ts).employee,
|
|
|
|
"employee_name": timesheets.get(ts).employee_name,
|
|
|
|
"timesheet": ts,
|
|
|
|
"total_billable_hours": total_billing_hours,
|
|
|
|
"total_hours": total_hours,
|
|
|
|
"amount": total_amount,
|
|
|
|
}
|
|
|
|
)
|
2019-03-11 17:43:44 +05:30
|
|
|
|
|
|
|
return data
|
|
|
|
|
2022-03-28 18:52:46 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
def get_timesheets(filters):
|
2019-03-12 13:12:31 +05:30
|
|
|
record_filters = [
|
2022-03-28 18:52:46 +05:30
|
|
|
["start_date", "<=", filters.to_date],
|
|
|
|
["end_date", ">=", filters.from_date],
|
|
|
|
["docstatus", "=", 1],
|
|
|
|
]
|
2019-03-12 13:12:31 +05:30
|
|
|
|
2019-03-11 18:17:22 +05:30
|
|
|
if "employee" in filters:
|
2019-03-12 13:12:31 +05:30
|
|
|
record_filters.append(["employee", "=", filters.employee])
|
2019-03-11 18:17:22 +05:30
|
|
|
|
2022-03-28 18:52:46 +05:30
|
|
|
timesheets = frappe.get_all(
|
|
|
|
"Timesheet", filters=record_filters, fields=["employee", "employee_name", "name"]
|
|
|
|
)
|
2019-05-02 18:11:08 +05:30
|
|
|
timesheet_map = frappe._dict()
|
|
|
|
for d in timesheets:
|
|
|
|
timesheet_map.setdefault(d.name, d)
|
2019-03-11 17:43:44 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
return timesheet_map
|
2019-03-11 18:17:22 +05:30
|
|
|
|
2022-03-28 18:52:46 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
def get_timesheet_details(filters, timesheet_list):
|
2022-03-28 18:52:46 +05:30
|
|
|
timesheet_details_filter = {"parent": ["in", timesheet_list]}
|
2019-03-11 18:17:22 +05:30
|
|
|
|
|
|
|
if "project" in filters:
|
|
|
|
timesheet_details_filter["project"] = filters.project
|
2019-03-11 17:43:44 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
timesheet_details = frappe.get_all(
|
2019-03-11 18:17:22 +05:30
|
|
|
"Timesheet Detail",
|
2022-03-28 18:52:46 +05:30
|
|
|
filters=timesheet_details_filter,
|
|
|
|
fields=[
|
|
|
|
"from_time",
|
|
|
|
"to_time",
|
|
|
|
"hours",
|
|
|
|
"is_billable",
|
|
|
|
"billing_hours",
|
|
|
|
"billing_rate",
|
|
|
|
"parent",
|
|
|
|
],
|
2019-05-02 18:11:08 +05:30
|
|
|
)
|
|
|
|
|
|
|
|
timesheet_details_map = frappe._dict()
|
|
|
|
for d in timesheet_details:
|
|
|
|
timesheet_details_map.setdefault(d.parent, []).append(d)
|
|
|
|
|
|
|
|
return timesheet_details_map
|
|
|
|
|
2022-03-28 18:52:46 +05:30
|
|
|
|
2019-05-02 18:11:08 +05:30
|
|
|
def get_billable_and_total_duration(activity, start_time, end_time):
|
2020-10-13 18:54:54 +05:30
|
|
|
precision = frappe.get_precision("Timesheet Detail", "hours")
|
2019-05-02 18:11:08 +05:30
|
|
|
activity_duration = time_diff_in_hours(end_time, start_time)
|
|
|
|
billing_duration = 0.0
|
2021-05-20 23:43:19 +05:30
|
|
|
if activity.is_billable:
|
2019-05-02 18:11:08 +05:30
|
|
|
billing_duration = activity.billing_hours
|
|
|
|
if activity_duration != activity.billing_hours:
|
|
|
|
billing_duration = activity_duration * activity.billing_hours / activity.hours
|
|
|
|
|
2021-08-19 13:41:10 +05:30
|
|
|
return flt(activity_duration, precision), flt(billing_duration, precision)
|