mirror of
https://github.com/frappe/erpnext.git
synced 2026-05-14 02:31:21 +00:00
Merge pull request #33600 from frappe/version-13-hotfix
chore: release v13
This commit is contained in:
4
.github/workflows/release.yml
vendored
4
.github/workflows/release.yml
vendored
@@ -13,10 +13,10 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- name: Setup Node.js v14
|
- name: Setup Node.js
|
||||||
uses: actions/setup-node@v2
|
uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: 14
|
node-version: 18
|
||||||
- name: Setup dependencies
|
- name: Setup dependencies
|
||||||
run: |
|
run: |
|
||||||
npm install @semantic-release/git @semantic-release/exec --no-save
|
npm install @semantic-release/git @semantic-release/exec --no-save
|
||||||
|
|||||||
@@ -102,7 +102,7 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-06-29 18:27:02.832979",
|
"modified": "2022-12-28 16:35:34.377575",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "CRM",
|
"module": "CRM",
|
||||||
"name": "Appointment",
|
"name": "Appointment",
|
||||||
@@ -121,16 +121,6 @@
|
|||||||
"share": 1,
|
"share": 1,
|
||||||
"write": 1
|
"write": 1
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"create": 1,
|
|
||||||
"email": 1,
|
|
||||||
"export": 1,
|
|
||||||
"print": 1,
|
|
||||||
"read": 1,
|
|
||||||
"report": 1,
|
|
||||||
"role": "Guest",
|
|
||||||
"share": 1
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"create": 1,
|
"create": 1,
|
||||||
"delete": 1,
|
"delete": 1,
|
||||||
|
|||||||
@@ -5,7 +5,9 @@
|
|||||||
from collections import Counter
|
from collections import Counter
|
||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
|
import frappe.share
|
||||||
from frappe import _
|
from frappe import _
|
||||||
|
from frappe.desk.form.assign_to import add as add_assignment
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.utils import get_url, getdate
|
from frappe.utils import get_url, getdate
|
||||||
from frappe.utils.verified_command import get_signed_params
|
from frappe.utils.verified_command import get_signed_params
|
||||||
@@ -118,21 +120,18 @@ class Appointment(Document):
|
|||||||
self.party = lead.name
|
self.party = lead.name
|
||||||
|
|
||||||
def auto_assign(self):
|
def auto_assign(self):
|
||||||
from frappe.desk.form.assign_to import add as add_assignemnt
|
|
||||||
|
|
||||||
existing_assignee = self.get_assignee_from_latest_opportunity()
|
existing_assignee = self.get_assignee_from_latest_opportunity()
|
||||||
if existing_assignee:
|
if existing_assignee:
|
||||||
# If the latest opportunity is assigned to someone
|
# If the latest opportunity is assigned to someone
|
||||||
# Assign the appointment to the same
|
# Assign the appointment to the same
|
||||||
add_assignemnt({"doctype": self.doctype, "name": self.name, "assign_to": [existing_assignee]})
|
self.assign_agent(existing_assignee)
|
||||||
return
|
return
|
||||||
if self._assign:
|
if self._assign:
|
||||||
return
|
return
|
||||||
available_agents = _get_agents_sorted_by_asc_workload(getdate(self.scheduled_time))
|
available_agents = _get_agents_sorted_by_asc_workload(getdate(self.scheduled_time))
|
||||||
for agent in available_agents:
|
for agent in available_agents:
|
||||||
if _check_agent_availability(agent, self.scheduled_time):
|
if _check_agent_availability(agent, self.scheduled_time):
|
||||||
agent = agent[0]
|
self.assign_agent(agent[0])
|
||||||
add_assignemnt({"doctype": self.doctype, "name": self.name, "assign_to": [agent]})
|
|
||||||
break
|
break
|
||||||
|
|
||||||
def get_assignee_from_latest_opportunity(self):
|
def get_assignee_from_latest_opportunity(self):
|
||||||
@@ -187,9 +186,15 @@ class Appointment(Document):
|
|||||||
params = {"email": self.customer_email, "appointment": self.name}
|
params = {"email": self.customer_email, "appointment": self.name}
|
||||||
return get_url(verify_route + "?" + get_signed_params(params))
|
return get_url(verify_route + "?" + get_signed_params(params))
|
||||||
|
|
||||||
|
def assign_agent(self, agent):
|
||||||
|
if not frappe.has_permission(doc=self, user=agent):
|
||||||
|
frappe.share.add(self.doctype, self.name, agent, flags={"ignore_share_permission": True})
|
||||||
|
|
||||||
|
add_assignment({"doctype": self.doctype, "name": self.name, "assign_to": [agent]})
|
||||||
|
|
||||||
|
|
||||||
def _get_agents_sorted_by_asc_workload(date):
|
def _get_agents_sorted_by_asc_workload(date):
|
||||||
appointments = frappe.db.get_list("Appointment", fields="*")
|
appointments = frappe.get_all("Appointment", fields="*")
|
||||||
agent_list = _get_agent_list_as_strings()
|
agent_list = _get_agent_list_as_strings()
|
||||||
if not appointments:
|
if not appointments:
|
||||||
return agent_list
|
return agent_list
|
||||||
@@ -214,7 +219,7 @@ def _get_agent_list_as_strings():
|
|||||||
|
|
||||||
|
|
||||||
def _check_agent_availability(agent_email, scheduled_time):
|
def _check_agent_availability(agent_email, scheduled_time):
|
||||||
appointemnts_at_scheduled_time = frappe.get_list(
|
appointemnts_at_scheduled_time = frappe.get_all(
|
||||||
"Appointment", filters={"scheduled_time": scheduled_time}
|
"Appointment", filters={"scheduled_time": scheduled_time}
|
||||||
)
|
)
|
||||||
for appointment in appointemnts_at_scheduled_time:
|
for appointment in appointemnts_at_scheduled_time:
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
{
|
{
|
||||||
|
"actions": [],
|
||||||
"creation": "2019-08-27 10:56:48.309824",
|
"creation": "2019-08-27 10:56:48.309824",
|
||||||
"doctype": "DocType",
|
"doctype": "DocType",
|
||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
@@ -101,7 +102,8 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"issingle": 1,
|
"issingle": 1,
|
||||||
"modified": "2019-11-26 12:14:17.669366",
|
"links": [],
|
||||||
|
"modified": "2022-12-28 16:41:28.773090",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "CRM",
|
"module": "CRM",
|
||||||
"name": "Appointment Booking Settings",
|
"name": "Appointment Booking Settings",
|
||||||
@@ -117,13 +119,6 @@
|
|||||||
"share": 1,
|
"share": 1,
|
||||||
"write": 1
|
"write": 1
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"email": 1,
|
|
||||||
"print": 1,
|
|
||||||
"read": 1,
|
|
||||||
"role": "Guest",
|
|
||||||
"share": 1
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"create": 1,
|
"create": 1,
|
||||||
"email": 1,
|
"email": 1,
|
||||||
|
|||||||
@@ -111,34 +111,30 @@ def get_data(filters: Filters) -> List:
|
|||||||
employee.leave_approver
|
employee.leave_approver
|
||||||
)
|
)
|
||||||
|
|
||||||
if (
|
if len(active_employees) > 1:
|
||||||
(leave_approvers and len(leave_approvers) and user in leave_approvers)
|
row = frappe._dict()
|
||||||
or (user in ["Administrator", employee.user_id])
|
|
||||||
or ("HR Manager" in frappe.get_roles(user))
|
|
||||||
):
|
|
||||||
if len(active_employees) > 1:
|
|
||||||
row = frappe._dict()
|
|
||||||
row.employee = employee.name
|
|
||||||
row.employee_name = employee.employee_name
|
|
||||||
|
|
||||||
leaves_taken = (
|
row.employee = employee.name
|
||||||
get_leaves_for_period(employee.name, leave_type, filters.from_date, filters.to_date) * -1
|
row.employee_name = employee.employee_name
|
||||||
)
|
|
||||||
|
|
||||||
new_allocation, expired_leaves, carry_forwarded_leaves = get_allocated_and_expired_leaves(
|
leaves_taken = (
|
||||||
filters.from_date, filters.to_date, employee.name, leave_type
|
get_leaves_for_period(employee.name, leave_type, filters.from_date, filters.to_date) * -1
|
||||||
)
|
)
|
||||||
opening = get_opening_balance(employee.name, leave_type, filters, carry_forwarded_leaves)
|
|
||||||
|
|
||||||
row.leaves_allocated = new_allocation
|
new_allocation, expired_leaves, carry_forwarded_leaves = get_allocated_and_expired_leaves(
|
||||||
row.leaves_expired = expired_leaves
|
filters.from_date, filters.to_date, employee.name, leave_type
|
||||||
row.opening_balance = opening
|
)
|
||||||
row.leaves_taken = leaves_taken
|
opening = get_opening_balance(employee.name, leave_type, filters, carry_forwarded_leaves)
|
||||||
|
|
||||||
# not be shown on the basis of days left it create in user mind for carry_forward leave
|
row.leaves_allocated = new_allocation
|
||||||
row.closing_balance = new_allocation + opening - (row.leaves_expired + leaves_taken)
|
row.leaves_expired = expired_leaves
|
||||||
row.indent = 1
|
row.opening_balance = opening
|
||||||
data.append(row)
|
row.leaves_taken = leaves_taken
|
||||||
|
|
||||||
|
# not be shown on the basis of days left it create in user mind for carry_forward leave
|
||||||
|
row.closing_balance = new_allocation + opening - (row.leaves_expired + leaves_taken)
|
||||||
|
row.indent = 1
|
||||||
|
data.append(row)
|
||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
||||||
|
|||||||
@@ -65,21 +65,16 @@ def get_data(filters, leave_types):
|
|||||||
if employee.leave_approver:
|
if employee.leave_approver:
|
||||||
leave_approvers.append(employee.leave_approver)
|
leave_approvers.append(employee.leave_approver)
|
||||||
|
|
||||||
if (
|
row = [employee.name, employee.employee_name, employee.department]
|
||||||
(len(leave_approvers) and user in leave_approvers)
|
available_leave = get_leave_details(employee.name, filters.date)
|
||||||
or (user in ["Administrator", employee.user_id])
|
for leave_type in leave_types:
|
||||||
or ("HR Manager" in frappe.get_roles(user))
|
remaining = 0
|
||||||
):
|
if leave_type in available_leave["leave_allocation"]:
|
||||||
row = [employee.name, employee.employee_name, employee.department]
|
# opening balance
|
||||||
available_leave = get_leave_details(employee.name, filters.date)
|
remaining = available_leave["leave_allocation"][leave_type]["remaining_leaves"]
|
||||||
for leave_type in leave_types:
|
|
||||||
remaining = 0
|
|
||||||
if leave_type in available_leave["leave_allocation"]:
|
|
||||||
# opening balance
|
|
||||||
remaining = available_leave["leave_allocation"][leave_type]["remaining_leaves"]
|
|
||||||
|
|
||||||
row += [remaining]
|
row += [remaining]
|
||||||
|
|
||||||
data.append(row)
|
data.append(row)
|
||||||
|
|
||||||
return data
|
return data
|
||||||
|
|||||||
@@ -123,6 +123,7 @@
|
|||||||
"fieldname": "route",
|
"fieldname": "route",
|
||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"label": "Route",
|
"label": "Route",
|
||||||
|
"no_copy": 1,
|
||||||
"unique": 1
|
"unique": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -232,11 +233,10 @@
|
|||||||
"is_tree": 1,
|
"is_tree": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"max_attachments": 3,
|
"max_attachments": 3,
|
||||||
"modified": "2022-03-09 12:27:11.055782",
|
"modified": "2023-01-05 12:21:30.458628",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Setup",
|
"module": "Setup",
|
||||||
"name": "Item Group",
|
"name": "Item Group",
|
||||||
"name_case": "Title Case",
|
|
||||||
"naming_rule": "By fieldname",
|
"naming_rule": "By fieldname",
|
||||||
"nsm_parent_field": "parent_item_group",
|
"nsm_parent_field": "parent_item_group",
|
||||||
"owner": "Administrator",
|
"owner": "Administrator",
|
||||||
|
|||||||
@@ -83,8 +83,8 @@ def reset_packing_list(doc):
|
|||||||
# 1. items were deleted
|
# 1. items were deleted
|
||||||
# 2. if bundle item replaced by another item (same no. of items but different items)
|
# 2. if bundle item replaced by another item (same no. of items but different items)
|
||||||
# we maintain list to track recurring item rows as well
|
# we maintain list to track recurring item rows as well
|
||||||
items_before_save = [item.item_code for item in doc_before_save.get("items")]
|
items_before_save = [(item.name, item.item_code) for item in doc_before_save.get("items")]
|
||||||
items_after_save = [item.item_code for item in doc.get("items")]
|
items_after_save = [(item.name, item.item_code) for item in doc.get("items")]
|
||||||
reset_table = items_before_save != items_after_save
|
reset_table = items_before_save != items_after_save
|
||||||
else:
|
else:
|
||||||
# reset: if via Update Items OR
|
# reset: if via Update Items OR
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ frappe.ready(async () => {
|
|||||||
initialise_select_date();
|
initialise_select_date();
|
||||||
})
|
})
|
||||||
|
|
||||||
window.holiday_list = [];
|
|
||||||
|
|
||||||
async function initialise_select_date() {
|
async function initialise_select_date() {
|
||||||
navigate_to_page(1);
|
navigate_to_page(1);
|
||||||
@@ -20,7 +19,6 @@ async function get_global_variables() {
|
|||||||
window.timezones = (await frappe.call({
|
window.timezones = (await frappe.call({
|
||||||
method:'erpnext.www.book_appointment.index.get_timezones'
|
method:'erpnext.www.book_appointment.index.get_timezones'
|
||||||
})).message;
|
})).message;
|
||||||
window.holiday_list = window.appointment_settings.holiday_list;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function setup_timezone_selector() {
|
function setup_timezone_selector() {
|
||||||
|
|||||||
@@ -26,8 +26,12 @@ def get_context(context):
|
|||||||
|
|
||||||
@frappe.whitelist(allow_guest=True)
|
@frappe.whitelist(allow_guest=True)
|
||||||
def get_appointment_settings():
|
def get_appointment_settings():
|
||||||
settings = frappe.get_doc("Appointment Booking Settings")
|
settings = frappe.get_cached_value(
|
||||||
settings.holiday_list = frappe.get_doc("Holiday List", settings.holiday_list)
|
"Appointment Booking Settings",
|
||||||
|
None,
|
||||||
|
["advance_booking_days", "appointment_duration", "success_redirect_url"],
|
||||||
|
as_dict=True,
|
||||||
|
)
|
||||||
return settings
|
return settings
|
||||||
|
|
||||||
|
|
||||||
@@ -106,7 +110,7 @@ def create_appointment(date, time, tz, contact):
|
|||||||
appointment.customer_details = contact.get("notes", None)
|
appointment.customer_details = contact.get("notes", None)
|
||||||
appointment.customer_email = contact.get("email", None)
|
appointment.customer_email = contact.get("email", None)
|
||||||
appointment.status = "Open"
|
appointment.status = "Open"
|
||||||
appointment.insert()
|
appointment.insert(ignore_permissions=True)
|
||||||
return appointment
|
return appointment
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import frappe
|
|||||||
from frappe.utils.verified_command import verify_request
|
from frappe.utils.verified_command import verify_request
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist(allow_guest=True)
|
|
||||||
def get_context(context):
|
def get_context(context):
|
||||||
if not verify_request():
|
if not verify_request():
|
||||||
context.success = False
|
context.success = False
|
||||||
|
|||||||
Reference in New Issue
Block a user