// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors // License: GNU General Public License v3. See license.txt frappe.provide("erpnext"); frappe.provide("erpnext.utils"); frappe.provide("erpnext.stock.utils"); $.extend(erpnext, { get_currency: function (company) { if (!company && cur_frm) company = cur_frm.doc.company; if (company) return frappe.get_doc(":Company", company)?.default_currency || frappe.boot.sysdefaults.currency; else return frappe.boot.sysdefaults.currency; }, get_presentation_currency_list: () => { const docs = frappe.boot.docs; let currency_list = docs.filter((d) => d.doctype === ":Currency").map((d) => d.name); currency_list.unshift(""); return currency_list; }, toggle_naming_series: function () { if ( cur_frm && cur_frm.fields_dict.naming_series && cur_frm.meta.naming_rule == 'By "Naming Series" field' ) { cur_frm.toggle_display("naming_series", cur_frm.doc.__islocal ? true : false); } }, hide_company: function (frm) { if (frm?.fields_dict.company) { var companies = Object.keys(locals[":Company"] || {}); if (companies.length === 1) { if (!frm.doc.company) frm.set_value("company", companies[0]); frm.toggle_display("company", false); } else if (erpnext.last_selected_company) { if (!frm.doc.company) frm.set_value("company", erpnext.last_selected_company); } } }, is_perpetual_inventory_enabled: function (company) { if (company) { let company_local = locals[":Company"] && locals[":Company"][company]; if (company_local) { return cint(company_local.enable_perpetual_inventory); } } }, stale_rate_allowed: () => { return cint(frappe.boot.sysdefaults.allow_stale); }, setup_serial_or_batch_no: function () { // Deprecated in v15 }, route_to_adjustment_jv: (args) => { frappe.model.with_doctype("Journal Entry", () => { // route to adjustment Journal Entry to handle Account Balance and Stock Value mismatch let journal_entry = frappe.model.get_new_doc("Journal Entry"); args.accounts.forEach((je_account) => { let child_row = frappe.model.add_child(journal_entry, "accounts"); child_row.account = je_account.account; child_row.debit_in_account_currency = je_account.debit_in_account_currency; child_row.credit_in_account_currency = je_account.credit_in_account_currency; child_row.party_type = ""; }); frappe.set_route("Form", "Journal Entry", journal_entry.name); }); }, route_to_pending_reposts: (args) => { frappe.set_route("List", "Repost Item Valuation", args); }, }); $.extend(erpnext.utils, { set_party_dashboard_indicators: function (frm) { if (frm.doc.__onload && frm.doc.__onload.dashboard_info) { var company_wise_info = frm.doc.__onload.dashboard_info; if (company_wise_info.length > 1) { company_wise_info.forEach(function (info) { erpnext.utils.add_indicator_for_multicompany(frm, info); }); } else if (company_wise_info.length === 1) { frm.dashboard.add_indicator( __("Annual Billing: {0}", [ format_currency( company_wise_info[0].billing_this_year, company_wise_info[0].currency ), ]), "blue" ); frm.dashboard.add_indicator( __("Total Unpaid: {0}", [ format_currency(company_wise_info[0].total_unpaid, company_wise_info[0].currency), ]), company_wise_info[0].total_unpaid ? "orange" : "green" ); if (company_wise_info[0].loyalty_points) { frm.dashboard.add_indicator( __("Loyalty Points: {0}", [company_wise_info[0].loyalty_points]), "blue" ); } } } }, view_serial_batch_nos: function (frm) { if (!frm.doc?.items) { return; } let bundle_ids = frm.doc.items.filter((d) => d.serial_and_batch_bundle); if (bundle_ids?.length) { frm.add_custom_button( __("Serial / Batch Nos"), () => { frappe.route_options = { voucher_no: frm.doc.name, voucher_type: frm.doc.doctype, from_date: frm.doc.posting_date || frm.doc.transaction_date, to_date: frm.doc.posting_date || frm.doc.transaction_date, company: frm.doc.company, }; frappe.set_route("query-report", "Serial and Batch Summary"); }, __("View") ); } }, add_indicator_for_multicompany: function (frm, info) { frm.dashboard.stats_area.show(); frm.dashboard.stats_area_row.addClass("flex"); frm.dashboard.stats_area_row.css("flex-wrap", "wrap"); var color = info.total_unpaid ? "orange" : "green"; var indicator = $( '
' + '
' + info.company + "
" + '" + '" + "
" ).appendTo(frm.dashboard.stats_area_row); if (info.loyalty_points) { $( '" ).appendTo(indicator); } return indicator; }, get_party_name: function (party_type) { var dict = { Customer: "customer_name", Supplier: "supplier_name", Employee: "employee_name", Member: "member_name", }; return dict[party_type]; }, copy_value_in_all_rows: function (doc, dt, dn, table_fieldname, fieldname) { var d = locals[dt][dn]; if (d[fieldname]) { var cl = doc[table_fieldname] || []; for (var i = 0; i < cl.length; i++) { if (!cl[i][fieldname]) cl[i][fieldname] = d[fieldname]; } } refresh_field(table_fieldname); }, get_terms: function (tc_name, doc, callback) { if (tc_name) { return frappe.call({ method: "erpnext.setup.doctype.terms_and_conditions.terms_and_conditions.get_terms_and_conditions", args: { template_name: tc_name, doc: doc, }, callback: function (r) { callback(r); }, }); } }, make_bank_account: function (doctype, docname) { frappe.call({ method: "erpnext.accounts.doctype.bank_account.bank_account.make_bank_account", args: { doctype: doctype, docname: docname, }, freeze: true, callback: function (r) { var doclist = frappe.model.sync(r.message); frappe.set_route("Form", doclist[0].doctype, doclist[0].name); }, }); }, add_dimensions: function (report_name, index) { let filters = frappe.query_reports[report_name].filters; frappe.call({ method: "erpnext.accounts.doctype.accounting_dimension.accounting_dimension.get_dimensions", callback: function (r) { let accounting_dimensions = r.message[0]; accounting_dimensions.forEach((dimension) => { let found = filters.some((el) => el.fieldname === dimension["fieldname"]); if (!found) { filters.splice(index, 0, { fieldname: dimension["fieldname"], label: __(dimension["label"]), fieldtype: "MultiSelectList", get_data: function (txt) { return frappe.db.get_link_options(dimension["document_type"], txt); }, }); } }); }, }); }, add_inventory_dimensions: function (report_name, index) { let filters = frappe.query_reports[report_name].filters; frappe.call({ method: "erpnext.stock.doctype.inventory_dimension.inventory_dimension.get_inventory_dimensions", callback: function (r) { if (r.message && r.message.length) { r.message.forEach((dimension) => { let existing_filter = filters.filter((el) => el.fieldname === dimension["fieldname"]); if (!existing_filter.length) { filters.splice(index, 0, { fieldname: dimension["fieldname"], label: __(dimension["doctype"]), fieldtype: "MultiSelectList", get_data: function (txt) { return frappe.db.get_link_options(dimension["doctype"], txt); }, }); } else { existing_filter[0]["fieldtype"] = "MultiSelectList"; existing_filter[0]["get_data"] = function (txt) { return frappe.db.get_link_options(dimension["doctype"], txt); }; } }); } }, }); }, make_subscription: function (doctype, docname) { frappe.call({ method: "frappe.automation.doctype.auto_repeat.auto_repeat.make_auto_repeat", args: { doctype: doctype, docname: docname, }, callback: function (r) { var doclist = frappe.model.sync(r.message); frappe.set_route("Form", doclist[0].doctype, doclist[0].name); }, }); }, make_pricing_rule: function (doctype, docname) { frappe.call({ method: "erpnext.accounts.doctype.pricing_rule.pricing_rule.make_pricing_rule", args: { doctype: doctype, docname: docname, }, callback: function (r) { var doclist = frappe.model.sync(r.message); frappe.set_route("Form", doclist[0].doctype, doclist[0].name); }, }); }, /** * Checks if the first row of a given child table is empty * @param child_table - Child table Doctype * @return {Boolean} **/ first_row_is_empty: function (child_table) { if ($.isArray(child_table) && child_table.length > 0) { return !child_table[0].item_code; } return false; }, /** * Removes the first row of a child table if it is empty * @param {_Frm} frm - The current form * @param {String} child_table_name - The child table field name * @return {Boolean} **/ remove_empty_first_row: function (frm, child_table_name) { const rows = frm["doc"][child_table_name]; if (this.first_row_is_empty(rows)) { frm["doc"][child_table_name] = rows.splice(1); } return rows; }, get_tree_options: function (option) { // get valid options for tree based on user permission & locals dict let unscrub_option = frappe.model.unscrub(option); let user_permission = frappe.defaults.get_user_permissions(); let options; if (user_permission && user_permission[unscrub_option]) { options = user_permission[unscrub_option].map((perm) => perm.doc); } else { options = $.map(locals[`:${unscrub_option}`], function (c) { return c.name; }).sort(); } // filter unique values, as there may be multiple user permissions for any value return options.filter((value, index, self) => self.indexOf(value) === index); }, get_tree_default: function (option) { // set default for a field based on user permission let options = this.get_tree_options(option); if (options.includes(frappe.defaults.get_default(option))) { return frappe.defaults.get_default(option); } else { return options[0]; } }, overrides_parent_value_in_all_rows: function (doc, dt, dn, table_fieldname, fieldname, parent_fieldname) { if (doc[parent_fieldname]) { let cl = doc[table_fieldname] || []; for (let i = 0; i < cl.length; i++) { cl[i][fieldname] = doc[parent_fieldname]; } frappe.refresh_field(table_fieldname); } }, create_new_doc: function (doctype, update_fields) { frappe.model.with_doctype(doctype, function () { var new_doc = frappe.model.get_new_doc(doctype); for (let [key, value] of Object.entries(update_fields)) { new_doc[key] = value; } frappe.ui.form.make_quick_entry(doctype, null, null, new_doc); }); }, // check if payments app is installed on site, if not warn user. check_payments_app: () => { if (frappe.boot.versions && !frappe.boot.versions.payments) { const marketplace_link = 'Marketplace'; const github_link = 'GitHub'; const msg = __("payments app is not installed. Please install it from {0} or {1}", [ marketplace_link, github_link, ]); frappe.msgprint(msg); } }, pick_serial_and_batch_bundle(frm, cdt, cdn, type_of_transaction, warehouse_field) { let item_row = frappe.get_doc(cdt, cdn); item_row.type_of_transaction = type_of_transaction; frappe.db.get_value("Item", item_row.item_code, ["has_batch_no", "has_serial_no"]).then((r) => { item_row.has_batch_no = r.message.has_batch_no; item_row.has_serial_no = r.message.has_serial_no; new erpnext.SerialBatchPackageSelector(frm, item_row, (r) => { if (r) { let update_values = { serial_and_batch_bundle: r.name, qty: Math.abs(r.total_qty), }; if (!warehouse_field) { warehouse_field = "warehouse"; } if (r.warehouse) { update_values[warehouse_field] = r.warehouse; } frappe.model.set_value(item_row.doctype, item_row.name, update_values); } }); }); }, get_fiscal_year: function (date, with_dates = false, raise_on_missing = true) { if (!frappe.boot.setup_complete) { return; } if (!date) { date = frappe.datetime.get_today(); } let fiscal_year = ""; if ( frappe.boot.current_fiscal_year && date >= frappe.boot.current_fiscal_year[1] && date <= frappe.boot.current_fiscal_year[2] ) { if (with_dates) fiscal_year = frappe.boot.current_fiscal_year; else fiscal_year = frappe.boot.current_fiscal_year[0]; } else { frappe.call({ method: "erpnext.accounts.utils.get_fiscal_year", type: "GET", // make it cacheable args: { date: date, raise_on_missing: raise_on_missing, }, async: false, callback: function (r) { if (r.message) { if (with_dates) fiscal_year = r.message; else fiscal_year = r.message[0]; } }, }); } return fiscal_year; }, }); erpnext.utils.select_alternate_items = function (opts) { const frm = opts.frm; const warehouse_field = opts.warehouse_field || "warehouse"; const item_field = opts.item_field || "item_code"; this.data = []; const dialog = new frappe.ui.Dialog({ title: __("Select Alternate Item"), fields: [ { fieldtype: "Section Break", label: __("Items") }, { fieldname: "alternative_items", fieldtype: "Table", cannot_add_rows: true, in_place_edit: true, data: this.data, get_data: () => { return this.data; }, fields: [ { fieldtype: "Data", fieldname: "docname", hidden: 1, }, { fieldtype: "Link", fieldname: "item_code", options: "Item", in_list_view: 1, read_only: 1, label: __("Item Code"), }, { fieldtype: "Link", fieldname: "alternate_item", options: "Item", default: "", in_list_view: 1, label: __("Alternate Item"), onchange: function () { const item_code = this.get_value(); const warehouse = this.grid_row.on_grid_fields_dict.warehouse.get_value(); if (item_code && warehouse) { frappe.call({ method: "erpnext.stock.utils.get_latest_stock_qty", args: { item_code: item_code, warehouse: warehouse, }, callback: (r) => { this.grid_row.on_grid_fields_dict.actual_qty.set_value( r.message || 0 ); }, }); } }, get_query: (e) => { return { query: "erpnext.stock.doctype.item_alternative.item_alternative.get_alternative_items", filters: { item_code: e.item_code, }, }; }, }, { fieldtype: "Link", fieldname: "warehouse", options: "Warehouse", default: "", in_list_view: 1, label: __("Warehouse"), onchange: function () { const warehouse = this.get_value(); const item_code = this.grid_row.on_grid_fields_dict.item_code.get_value(); if (item_code && warehouse) { frappe.call({ method: "erpnext.stock.utils.get_latest_stock_qty", args: { item_code: item_code, warehouse: warehouse, }, callback: (r) => { this.grid_row.on_grid_fields_dict.actual_qty.set_value( r.message || 0 ); }, }); } }, }, { fieldtype: "Float", fieldname: "actual_qty", default: 0, read_only: 1, in_list_view: 1, label: __("Available Qty"), }, ], }, ], primary_action: function () { const args = this.get_values()["alternative_items"]; const alternative_items = args.filter((d) => { if (d.alternate_item && d.item_code != d.alternate_item) { return true; } }); alternative_items.forEach((d) => { let row = frappe.get_doc(opts.child_doctype, d.docname); let qty = null; if (row.doctype === "Work Order Item") { qty = row.required_qty; } else { qty = row.qty; } row[item_field] = d.alternate_item; frappe.model.set_value(row.doctype, row.name, "qty", qty); frappe.model.set_value(row.doctype, row.name, opts.original_item_field, d.item_code); frm.trigger(item_field, row.doctype, row.name); }); refresh_field(opts.child_docname); this.hide(); }, primary_action_label: __("Update"), }); frm.doc[opts.child_docname].forEach((d) => { if (!opts.condition || opts.condition(d)) { dialog.fields_dict.alternative_items.df.data.push({ docname: d.name, item_code: d[item_field], warehouse: d[warehouse_field], actual_qty: d.actual_qty, }); } }); this.data = dialog.fields_dict.alternative_items.df.data; dialog.fields_dict.alternative_items.grid.refresh(); dialog.show(); }; erpnext.utils.update_child_items = function (opts) { const frm = opts.frm; const cannot_add_row = typeof opts.cannot_add_row === "undefined" ? true : opts.cannot_add_row; const child_docname = typeof opts.cannot_add_row === "undefined" ? "items" : opts.child_docname; const child_meta = frappe.get_meta(`${frm.doc.doctype} Item`); const has_reserved_stock = opts.has_reserved_stock ? true : false; const get_precision = (fieldname) => child_meta.fields.find((f) => f.fieldname == fieldname).precision; this.data = frm.doc[opts.child_docname].map((d) => { return { docname: d.name, name: d.name, item_code: d.item_code, delivery_date: d.delivery_date, schedule_date: d.schedule_date, conversion_factor: d.conversion_factor, qty: d.qty, rate: d.rate, uom: d.uom, fg_item: d.fg_item, fg_item_qty: d.fg_item_qty, }; }); const fields = [ { fieldtype: "Data", fieldname: "docname", read_only: 1, hidden: 1, }, { fieldtype: "Link", fieldname: "item_code", options: "Item", in_list_view: 1, read_only: 0, disabled: 0, label: __("Item Code"), get_query: function () { let filters; if (frm.doc.doctype == "Sales Order") { filters = { is_sales_item: 1 }; } else if (frm.doc.doctype == "Purchase Order") { if (frm.doc.is_subcontracted) { if (frm.doc.is_old_subcontracting_flow) { filters = { is_sub_contracted_item: 1 }; } else { filters = { is_stock_item: 0 }; } } else { filters = { is_purchase_item: 1 }; } } return { query: "erpnext.controllers.queries.item_query", filters: filters, }; }, onchange: function () { const me = this; frm.call({ method: "erpnext.stock.get_item_details.get_item_details", args: { doc: frm.doc, ctx: { item_code: this.value, set_warehouse: frm.doc.set_warehouse, customer: frm.doc.customer || frm.doc.party_name, quotation_to: frm.doc.quotation_to, supplier: frm.doc.supplier, currency: frm.doc.currency, is_internal_supplier: frm.doc.is_internal_supplier, is_internal_customer: frm.doc.is_internal_customer, conversion_rate: frm.doc.conversion_rate, price_list: frm.doc.selling_price_list || frm.doc.buying_price_list, price_list_currency: frm.doc.price_list_currency, plc_conversion_rate: frm.doc.plc_conversion_rate, company: frm.doc.company, order_type: frm.doc.order_type, is_pos: cint(frm.doc.is_pos), is_return: cint(frm.doc.is_return), is_subcontracted: frm.doc.is_subcontracted, ignore_pricing_rule: frm.doc.ignore_pricing_rule, doctype: frm.doc.doctype, name: frm.doc.name, qty: me.doc.qty || 1, uom: me.doc.uom, pos_profile: cint(frm.doc.is_pos) ? frm.doc.pos_profile : "", tax_category: frm.doc.tax_category, child_doctype: frm.doc.doctype + " Item", is_old_subcontracting_flow: frm.doc.is_old_subcontracting_flow, }, }, callback: function (r) { if (r.message) { const { qty, price_list_rate: rate, uom, conversion_factor } = r.message; const row = dialog.fields_dict.trans_items.df.data.find( (doc) => doc.idx == me.doc.idx ); if (row) { Object.assign(row, { conversion_factor: me.doc.conversion_factor || conversion_factor, uom: me.doc.uom || uom, qty: me.doc.qty || qty, rate: me.doc.rate || rate, }); dialog.fields_dict.trans_items.grid.refresh(); } } }, }); }, }, { fieldtype: "Link", fieldname: "uom", options: "UOM", read_only: 0, label: __("UOM"), reqd: 1, onchange: function () { frappe.call({ method: "erpnext.stock.get_item_details.get_conversion_factor", args: { item_code: this.doc.item_code, uom: this.value }, callback: (r) => { if (!r.exc) { if (this.doc.conversion_factor == r.message.conversion_factor) return; const docname = this.doc.docname; dialog.fields_dict.trans_items.df.data.some((doc) => { if (doc.docname == docname) { doc.conversion_factor = r.message.conversion_factor; dialog.fields_dict.trans_items.grid.refresh(); return true; } }); } }, }); }, }, { fieldtype: "Float", fieldname: "qty", default: 0, read_only: 0, in_list_view: 1, label: __("Qty"), precision: get_precision("qty"), }, { fieldtype: "Currency", fieldname: "rate", options: "currency", default: 0, read_only: 0, in_list_view: 1, label: __("Rate"), precision: get_precision("rate"), }, ]; if (frm.doc.doctype == "Sales Order" || frm.doc.doctype == "Purchase Order") { fields.splice(2, 0, { fieldtype: "Date", fieldname: frm.doc.doctype == "Sales Order" ? "delivery_date" : "schedule_date", in_list_view: 1, label: frm.doc.doctype == "Sales Order" ? __("Delivery Date") : __("Reqd by date"), default: frm.doc.doctype == "Sales Order" ? frm.doc.delivery_date : frm.doc.schedule_date, reqd: 1, }); fields.splice(3, 0, { fieldtype: "Float", fieldname: "conversion_factor", label: __("Conversion Factor"), precision: get_precision("conversion_factor"), }); } if ( frm.doc.doctype == "Purchase Order" && frm.doc.is_subcontracted && !frm.doc.is_old_subcontracting_flow ) { fields.push( { fieldtype: "Link", fieldname: "fg_item", options: "Item", reqd: 1, in_list_view: 0, read_only: 0, disabled: 0, label: __("Finished Good Item"), get_query: () => { return { filters: { is_stock_item: 1, is_sub_contracted_item: 1, default_bom: ["!=", ""], }, }; }, }, { fieldtype: "Float", fieldname: "fg_item_qty", reqd: 1, default: 0, read_only: 0, in_list_view: 0, label: __("Finished Good Item Qty"), precision: get_precision("fg_item_qty"), } ); } let dialog = new frappe.ui.Dialog({ title: __("Update Items"), size: "extra-large", fields: [ { fieldname: "trans_items", fieldtype: "Table", label: "Items", cannot_add_rows: cannot_add_row, in_place_edit: false, reqd: 1, data: this.data, get_data: () => { return this.data; }, fields: fields, }, ], primary_action: function () { if (frm.doctype == "Sales Order" && has_reserved_stock) { this.hide(); frappe.confirm( __( "The reserved stock will be released when you update items. Are you certain you wish to proceed?" ), () => this.update_items() ); } else { this.update_items(); } }, update_items: function () { const trans_items = this.get_values()["trans_items"].filter((item) => !!item.item_code); frappe.call({ method: "erpnext.controllers.accounts_controller.update_child_qty_rate", freeze: true, args: { parent_doctype: frm.doc.doctype, trans_items: trans_items, parent_doctype_name: frm.doc.name, child_docname: child_docname, }, callback: function () { frm.reload_doc(); }, }); this.hide(); refresh_field("items"); }, primary_action_label: __("Update"), }); dialog.show(); }; erpnext.utils.map_current_doc = function (opts) { function _map() { if ($.isArray(cur_frm.doc.items) && cur_frm.doc.items.length > 0) { // remove first item row if empty if (!cur_frm.doc.items[0].item_code) { cur_frm.doc.items = cur_frm.doc.items.splice(1); } // find the doctype of the items table var items_doctype = frappe.meta.get_docfield(cur_frm.doctype, "items").options; // find the link fieldname from items table for the given // source_doctype var link_fieldname = null; frappe.get_meta(items_doctype).fields.forEach(function (d) { if (d.options === opts.source_doctype) link_fieldname = d.fieldname; }); // search in existing items if the source_name is already set and full qty fetched var already_set = false; var item_qty_map = {}; $.each(cur_frm.doc.items, function (i, d) { opts.source_name.forEach(function (src) { if (d[link_fieldname] == src) { already_set = true; if (item_qty_map[d.item_code]) item_qty_map[d.item_code] += flt(d.qty); else item_qty_map[d.item_code] = flt(d.qty); } }); }); if (already_set) { opts.source_name.forEach(function (src) { frappe.model.with_doc(opts.source_doctype, src, function (r) { var source_doc = frappe.model.get_doc(opts.source_doctype, src); $.each(source_doc.items || [], function (i, row) { if (row.qty > flt(item_qty_map[row.item_code])) { already_set = false; return false; } }); }); if (already_set) { frappe.msgprint( __("You have already selected items from {0} {1}", [opts.source_doctype, src]) ); return; } }); } } return frappe.call({ // Sometimes we hit the limit for URL length of a GET request // as we send the full target_doc. Hence this is a POST request. type: "POST", method: "frappe.model.mapper.map_docs", args: { method: opts.method, source_names: opts.source_name, target_doc: cur_frm.doc, args: opts.args, }, freeze: true, freeze_message: __("Mapping {0} ...", [opts.source_doctype]), callback: function (r) { if (!r.exc) { frappe.model.sync(r.message); cur_frm.dirty(); cur_frm.refresh(); } }, }); } let query_args = {}; if (opts.get_query_filters) { query_args.filters = opts.get_query_filters; } if (opts.get_query_method) { query_args.query = opts.get_query_method; } if (query_args.filters || query_args.query) { opts.get_query = () => query_args; } if (opts.source_doctype) { let data_fields = []; if (["Purchase Receipt", "Delivery Note"].includes(opts.source_doctype)) { let target_meta = frappe.get_meta(cur_frm.doc.doctype); if (target_meta.fields.find((f) => f.fieldname === "taxes")) { data_fields.push({ fieldname: "merge_taxes", fieldtype: "Check", label: __("Merge taxes from multiple documents"), }); } } const d = new frappe.ui.form.MultiSelectDialog({ doctype: opts.source_doctype, target: opts.target, date_field: opts.date_field || undefined, setters: opts.setters, read_only_setters: opts.read_only_setters, data_fields: data_fields, get_query: opts.get_query, add_filters_group: 1, allow_child_item_selection: opts.allow_child_item_selection, child_fieldname: opts.child_fieldname, child_columns: opts.child_columns, size: opts.size, action: function (selections, args) { let values = selections; if (values.length === 0) { frappe.msgprint(__("Please select {0}", [opts.source_doctype])); return; } if (values.constructor === Array) { opts.source_name = [...new Set(values)]; } else { opts.source_name = values; } if ( opts.allow_child_item_selection || ["Purchase Receipt", "Delivery Note"].includes(opts.source_doctype) ) { // args contains filtered child docnames opts.args = args; } d.dialog.hide(); _map(); }, }); return d; } if (opts.source_name) { opts.source_name = [opts.source_name]; _map(); } }; frappe.form.link_formatters["Item"] = function (value, doc, df) { return add_link_title(value, doc, df, "item_name"); }; frappe.form.link_formatters["Employee"] = function (value, doc, df) { return add_link_title(value, doc, df, "employee_name"); }; frappe.form.link_formatters["Project"] = function (value, doc, df) { return add_link_title(value, doc, df, "project_name"); }; /** * Add a title to a link value based on the provided document and field information. * * @param {string} value - The value to add a link title to. * @param {Object} doc - The document object. * @param {Object} df - The field object. * @param {string} title_field - The field name for the title. * @returns {string} - The link value with the added title. */ function add_link_title(value, doc, df, title_field) { if (doc && value && doc[title_field] && doc[title_field] !== value && doc[df.fieldname] === value) { return value + ": " + doc[title_field]; } else if (!value && doc.doctype && doc[title_field]) { return doc[title_field]; } else { return value; } } // add description on posting time $(document).on("app_ready", function () { if (!frappe.datetime.is_timezone_same()) { $.each( [ "Stock Reconciliation", "Stock Entry", "Stock Ledger Entry", "Delivery Note", "Purchase Receipt", "Sales Invoice", ], function (i, d) { frappe.ui.form.on(d, "onload", function (frm) { cur_frm.set_df_property("posting_time", "description", frappe.sys_defaults.time_zone); }); } ); } }); // Show SLA dashboard $(document).on("app_ready", function () { $.each(frappe.boot.service_level_agreement_doctypes, function (_i, d) { frappe.ui.form.on(d, { onload: function (frm) { if (!frm.doc.service_level_agreement) return; frappe.call({ method: "erpnext.support.doctype.service_level_agreement.service_level_agreement.get_service_level_agreement_filters", args: { doctype: frm.doc.doctype, name: frm.doc.service_level_agreement, customer: frm.doc.customer, }, callback: function (r) { if (r && r.message) { frm.set_query("priority", function () { return { filters: { name: ["in", r.message.priority], }, }; }); frm.set_query("service_level_agreement", function () { return { filters: { name: ["in", r.message.service_level_agreements], }, }; }); } }, }); }, refresh: function (frm) { if ( frm.doc.status !== "Closed" && frm.doc.service_level_agreement && ["First Response Due", "Resolution Due"].includes(frm.doc.agreement_status) ) { frappe.call({ method: "frappe.client.get", args: { doctype: "Service Level Agreement", name: frm.doc.service_level_agreement, }, callback: function (data) { let statuses = data.message.pause_sla_on; const hold_statuses = []; $.each(statuses, (_i, entry) => { hold_statuses.push(entry.status); }); if (hold_statuses.includes(frm.doc.status)) { frm.dashboard.clear_headline(); let message = { indicator: "orange", msg: __("SLA is on hold since {0}", [ moment(frm.doc.on_hold_since).fromNow(true), ]), }; frm.dashboard.set_headline_alert( '
' + '
' + '' + message.msg + " " + "
" + "
" ); } else { set_time_to_resolve_and_response(frm, data.message.apply_sla_for_resolution); } }, }); } else if (frm.doc.service_level_agreement) { frm.dashboard.clear_headline(); let agreement_status = frm.doc.agreement_status == "Fulfilled" ? { indicator: "green", msg: "Service Level Agreement has been fulfilled" } : { indicator: "red", msg: "Service Level Agreement Failed" }; frm.dashboard.set_headline_alert( '
' + '
' + ' " + "
" + "
" ); } }, }); }); }); function set_time_to_resolve_and_response(frm, apply_sla_for_resolution) { frm.dashboard.clear_headline(); let time_to_respond; if (!frm.doc.first_responded_on) { time_to_respond = get_time_left(frm.doc.response_by, frm.doc.agreement_status); } else { time_to_respond = get_status(frm.doc.response_by, frm.doc.first_responded_on); } let alert = `
Time to Respond: ${time_to_respond.diff_display}
`; if (apply_sla_for_resolution) { let time_to_resolve; if (!frm.doc.resolution_date) { time_to_resolve = get_time_left(frm.doc.resolution_by, frm.doc.agreement_status); } else { time_to_resolve = get_status(frm.doc.resolution_by, frm.doc.resolution_date); } alert += `
Time to Resolve: ${time_to_resolve.diff_display}
`; } alert += "
"; frm.dashboard.set_headline_alert(alert); } function get_time_left(timestamp, agreement_status) { const diff = moment(timestamp).diff(frappe.datetime.system_datetime(true)); const diff_display = diff >= 44500 ? moment.duration(diff).humanize() : "Failed"; let indicator = diff_display == "Failed" && agreement_status != "Fulfilled" ? "red" : "green"; return { diff_display: diff_display, indicator: indicator }; } function get_status(expected, actual) { const time_left = moment(expected).diff(moment(actual)); if (time_left >= 0) { return { diff_display: "Fulfilled", indicator: "green" }; } else { return { diff_display: "Failed", indicator: "red" }; } } function attach_selector_button(inner_text, append_loction, context, grid_row) { let $btn_div = $("
").css({ "margin-bottom": "10px", "margin-top": "10px" }).appendTo(append_loction); let $btn = $(``).appendTo($btn_div); $btn.on("click", function () { context.show_serial_batch_selector(grid_row.frm, grid_row.doc, "", "", true); }); } $.extend(erpnext.stock.utils, { set_item_details_using_barcode(frm, child_row, callback) { const barcode_scanner = new erpnext.utils.BarcodeScanner({ frm: frm }); barcode_scanner.scan_api_call(child_row.barcode, callback); }, get_serial_range(range_string, separator) { /* Return an array of serial numbers generated from a range string. Examples (using separator "::"): - "1::5" => ["1", "2", "3", "4", "5"] - "SN0009::12" => ["SN0009", "SN0010", "SN0011", "SN0012"] - "ABC//05::8" => ["ABC//05", "ABC//06", "ABC//07", "ABC//08"] */ if (!range_string) { return; } const [start_str, end_str] = range_string.trim().split(separator); if (!start_str || !end_str) { return; } const end_int = parseInt(end_str); const length_difference = start_str.length - end_str.length; const start_int = parseInt(start_str.substring(length_difference)); if (isNaN(start_int) || isNaN(end_int)) { return; } const serial_numbers = Array(end_int - start_int + 1) .fill(1) .map((x, y) => x + y) .map((x) => x + start_int - 1); return serial_numbers.map((val) => { return start_str.substring(0, length_difference) + val.toString().padStart(end_str.length, "0"); }); }, });