2023-07-23 12:07:21 +05:30
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
// License: GNU General Public License v3. See license.txt
frappe . provide ( "erpnext.selling" ) ;
erpnext . sales _common = {
setup _selling _controller : function ( ) {
erpnext . selling . SellingController = class SellingController extends erpnext . TransactionController {
setup ( ) {
super . setup ( ) ;
2023-07-23 22:44:18 +05:30
this . frm . email _field = "contact_email" ;
2023-07-23 12:07:21 +05:30
}
onload ( ) {
super . onload ( ) ;
this . setup _queries ( ) ;
this . frm . set _query ( 'shipping_rule' , function ( ) {
return {
filters : {
"shipping_rule_type" : "Selling"
}
} ;
} ) ;
}
setup _queries ( ) {
var me = this ;
$ . each ( [ [ "customer" , "customer" ] ,
[ "lead" , "lead" ] ] ,
function ( i , opts ) {
if ( me . frm . fields _dict [ opts [ 0 ] ] )
me . frm . set _query ( opts [ 0 ] , erpnext . queries [ opts [ 1 ] ] ) ;
} ) ;
me . frm . set _query ( 'contact_person' , erpnext . queries . contact _query ) ;
me . frm . set _query ( 'customer_address' , erpnext . queries . address _query ) ;
me . frm . set _query ( 'shipping_address_name' , erpnext . queries . address _query ) ;
me . frm . set _query ( 'dispatch_address_name' , erpnext . queries . dispatch _address _query ) ;
erpnext . accounts . dimensions . setup _dimension _filters ( me . frm , me . frm . doctype ) ;
if ( this . frm . fields _dict . selling _price _list ) {
this . frm . set _query ( "selling_price_list" , function ( ) {
return { filters : { selling : 1 } } ;
} ) ;
}
if ( this . frm . fields _dict . tc _name ) {
this . frm . set _query ( "tc_name" , function ( ) {
return { filters : { selling : 1 } } ;
} ) ;
}
if ( ! this . frm . fields _dict [ "items" ] ) {
return ;
}
if ( this . frm . fields _dict [ "items" ] . grid . get _field ( 'item_code' ) ) {
this . frm . set _query ( "item_code" , "items" , function ( ) {
return {
query : "erpnext.controllers.queries.item_query" ,
2023-07-23 22:44:18 +05:30
filters : { 'is_sales_item' : 1 , 'customer' : me . frm . doc . customer , 'has_variants' : 0 }
2023-07-23 12:07:21 +05:30
}
} ) ;
}
if ( this . frm . fields _dict [ "packed_items" ] &&
this . frm . fields _dict [ "packed_items" ] . grid . get _field ( 'batch_no' ) ) {
this . frm . set _query ( "batch_no" , "packed_items" , function ( doc , cdt , cdn ) {
return me . set _query _for _batch ( doc , cdt , cdn )
} ) ;
}
if ( this . frm . fields _dict [ "items" ] . grid . get _field ( 'item_code' ) ) {
this . frm . set _query ( "item_tax_template" , "items" , function ( doc , cdt , cdn ) {
return me . set _query _for _item _tax _template ( doc , cdt , cdn )
} ) ;
}
}
refresh ( ) {
super . refresh ( ) ;
frappe . dynamic _link = { doc : this . frm . doc , fieldname : 'customer' , doctype : 'Customer' }
this . frm . toggle _display ( "customer_name" ,
( this . frm . doc . customer _name && this . frm . doc . customer _name !== this . frm . doc . customer ) ) ;
this . toggle _editable _price _list _rate ( ) ;
}
customer ( ) {
var me = this ;
erpnext . utils . get _party _details ( this . frm , null , null , function ( ) {
me . apply _price _list ( ) ;
} ) ;
}
customer _address ( ) {
erpnext . utils . get _address _display ( this . frm , "customer_address" ) ;
erpnext . utils . set _taxes _from _address ( this . frm , "customer_address" , "customer_address" , "shipping_address_name" ) ;
}
shipping _address _name ( ) {
erpnext . utils . get _address _display ( this . frm , "shipping_address_name" , "shipping_address" ) ;
erpnext . utils . set _taxes _from _address ( this . frm , "shipping_address_name" , "customer_address" , "shipping_address_name" ) ;
}
dispatch _address _name ( ) {
erpnext . utils . get _address _display ( this . frm , "dispatch_address_name" , "dispatch_address" ) ;
}
sales _partner ( ) {
this . apply _pricing _rule ( ) ;
}
campaign ( ) {
this . apply _pricing _rule ( ) ;
}
selling _price _list ( ) {
this . apply _price _list ( ) ;
this . set _dynamic _labels ( ) ;
}
discount _percentage ( doc , cdt , cdn ) {
var item = frappe . get _doc ( cdt , cdn ) ;
item . discount _amount = 0.0 ;
this . apply _discount _on _item ( doc , cdt , cdn , 'discount_percentage' ) ;
}
discount _amount ( doc , cdt , cdn ) {
if ( doc . name === cdn ) {
return ;
}
var item = frappe . get _doc ( cdt , cdn ) ;
item . discount _percentage = 0.0 ;
this . apply _discount _on _item ( doc , cdt , cdn , 'discount_amount' ) ;
}
commission _rate ( ) {
this . calculate _commission ( ) ;
}
total _commission ( ) {
frappe . model . round _floats _in ( this . frm . doc , [ "amount_eligible_for_commission" , "total_commission" ] ) ;
const { amount _eligible _for _commission } = this . frm . doc ;
if ( ! amount _eligible _for _commission ) return ;
this . frm . set _value (
"commission_rate" , flt (
this . frm . doc . total _commission * 100.0 / amount _eligible _for _commission
)
) ;
}
allocated _percentage ( doc , cdt , cdn ) {
var sales _person = frappe . get _doc ( cdt , cdn ) ;
if ( sales _person . allocated _percentage ) {
sales _person . allocated _percentage = flt ( sales _person . allocated _percentage ,
precision ( "allocated_percentage" , sales _person ) ) ;
sales _person . allocated _amount = flt ( this . frm . doc . amount _eligible _for _commission *
sales _person . allocated _percentage / 100.0 ,
precision ( "allocated_amount" , sales _person ) ) ;
refresh _field ( [ "allocated_amount" ] , sales _person ) ;
this . calculate _incentive ( sales _person ) ;
refresh _field ( [ "allocated_percentage" , "allocated_amount" , "commission_rate" , "incentives" ] , sales _person . name ,
sales _person . parentfield ) ;
}
}
sales _person ( doc , cdt , cdn ) {
var row = frappe . get _doc ( cdt , cdn ) ;
this . calculate _incentive ( row ) ;
refresh _field ( "incentives" , row . name , row . parentfield ) ;
}
toggle _editable _price _list _rate ( ) {
var df = frappe . meta . get _docfield ( this . frm . doc . doctype + " Item" , "price_list_rate" , this . frm . doc . name ) ;
var editable _price _list _rate = cint ( frappe . defaults . get _default ( "editable_price_list_rate" ) ) ;
if ( df && editable _price _list _rate ) {
const parent _field = frappe . meta . get _parentfield ( this . frm . doc . doctype , this . frm . doc . doctype + " Item" ) ;
if ( ! this . frm . fields _dict [ parent _field ] ) return ;
this . frm . fields _dict [ parent _field ] . grid . update _docfield _property (
'price_list_rate' , 'read_only' , 0
) ;
}
}
calculate _commission ( ) {
if ( ! this . frm . fields _dict . commission _rate || this . frm . doc . docstatus === 1 ) return ;
if ( this . frm . doc . commission _rate > 100 ) {
this . frm . set _value ( "commission_rate" , 100 ) ;
frappe . throw ( ` ${ _ _ ( frappe . meta . get _label (
this . frm . doc . doctype , "commission_rate" , this . frm . doc . name
) ) } $ { _ _ ( "cannot be greater than 100" ) } ` );
}
this . frm . doc . amount _eligible _for _commission = this . frm . doc . items . reduce (
( sum , item ) => item . grant _commission ? sum + item . base _net _amount : sum , 0
)
this . frm . doc . total _commission = flt (
this . frm . doc . amount _eligible _for _commission * this . frm . doc . commission _rate / 100.0 ,
precision ( "total_commission" )
) ;
refresh _field ( [ "amount_eligible_for_commission" , "total_commission" ] ) ;
}
calculate _contribution ( ) {
var me = this ;
$ . each ( this . frm . doc . doctype . sales _team || [ ] , function ( i , sales _person ) {
frappe . model . round _floats _in ( sales _person ) ;
if ( ! sales _person . allocated _percentage ) return ;
sales _person . allocated _amount = flt (
me . frm . doc . amount _eligible _for _commission
* sales _person . allocated _percentage
/ 1 0 0 . 0 ,
precision ( "allocated_amount" , sales _person )
) ;
} ) ;
}
calculate _incentive ( row ) {
if ( row . allocated _amount )
{
row . incentives = flt (
row . allocated _amount * row . commission _rate / 100.0 ,
precision ( "incentives" , row ) ) ;
}
}
set _dynamic _labels ( ) {
super . set _dynamic _labels ( ) ;
this . set _product _bundle _help ( this . frm . doc ) ;
}
set _product _bundle _help ( doc ) {
2023-07-23 22:44:18 +05:30
if ( ! this . frm . fields _dict . packing _list ) return ;
2023-07-23 12:07:21 +05:30
if ( ( doc . packed _items || [ ] ) . length ) {
2023-07-23 22:44:18 +05:30
$ ( this . frm . fields _dict . packing _list . row . wrapper ) . toggle ( true ) ;
2023-07-23 12:07:21 +05:30
if ( in _list ( [ 'Delivery Note' , 'Sales Invoice' ] , doc . doctype ) ) {
var help _msg = "<div class='alert alert-warning'>" +
_ _ ( "For 'Product Bundle' items, Warehouse, Serial No and Batch No will be considered from the 'Packing List' table. If Warehouse and Batch No are same for all packing items for any 'Product Bundle' item, those values can be entered in the main Item table, values will be copied to 'Packing List' table." ) +
"</div>" ;
frappe . meta . get _docfield ( doc . doctype , 'product_bundle_help' , doc . name ) . options = help _msg ;
}
} else {
2023-07-23 22:44:18 +05:30
$ ( this . frm . fields _dict . packing _list . row . wrapper ) . toggle ( false ) ;
2023-07-23 12:07:21 +05:30
if ( in _list ( [ 'Delivery Note' , 'Sales Invoice' ] , doc . doctype ) ) {
frappe . meta . get _docfield ( doc . doctype , 'product_bundle_help' , doc . name ) . options = '' ;
}
}
refresh _field ( 'product_bundle_help' ) ;
}
company _address ( ) {
var me = this ;
if ( this . frm . doc . company _address ) {
frappe . call ( {
method : "frappe.contacts.doctype.address.address.get_address_display" ,
args : { "address_dict" : this . frm . doc . company _address } ,
callback : function ( r ) {
if ( r . message ) {
me . frm . set _value ( "company_address_display" , r . message )
}
}
} )
} else {
this . frm . set _value ( "company_address_display" , "" ) ;
}
}
conversion _factor ( doc , cdt , cdn , dont _fetch _price _list _rate ) {
super . conversion _factor ( doc , cdt , cdn , dont _fetch _price _list _rate ) ;
}
qty ( doc , cdt , cdn ) {
super . qty ( doc , cdt , cdn ) ;
}
pick _serial _and _batch ( doc , cdt , cdn ) {
let item = locals [ cdt ] [ cdn ] ;
let me = this ;
let path = "assets/erpnext/js/utils/serial_no_batch_selector.js" ;
frappe . db . get _value ( "Item" , item . item _code , [ "has_batch_no" , "has_serial_no" ] )
. then ( ( r ) => {
if ( r . message && ( r . message . has _batch _no || r . message . has _serial _no ) ) {
item . has _serial _no = r . message . has _serial _no ;
item . has _batch _no = r . message . has _batch _no ;
item . type _of _transaction = item . qty > 0 ? "Outward" : "Inward" ;
item . title = item . has _serial _no ?
_ _ ( "Select Serial No" ) : _ _ ( "Select Batch No" ) ;
if ( item . has _serial _no && item . has _batch _no ) {
item . title = _ _ ( "Select Serial and Batch" ) ;
}
frappe . require ( path , function ( ) {
new erpnext . SerialBatchPackageSelector (
me . frm , item , ( r ) => {
if ( r ) {
frappe . model . set _value ( item . doctype , item . name , {
"serial_and_batch_bundle" : r . name ,
"qty" : Math . abs ( r . total _qty )
} ) ;
}
}
) ;
} ) ;
}
} ) ;
}
update _auto _repeat _reference ( doc ) {
if ( doc . auto _repeat ) {
frappe . call ( {
method : "frappe.automation.doctype.auto_repeat.auto_repeat.update_reference" ,
args : {
docname : doc . auto _repeat ,
reference : doc . name
} ,
callback : function ( r ) {
if ( r . message == "success" ) {
frappe . show _alert ( { message : _ _ ( "Auto repeat document updated" ) , indicator : 'green' } ) ;
} else {
frappe . show _alert ( { message : _ _ ( "An error occurred during the update process" ) , indicator : 'red' } ) ;
}
}
} )
}
}
project ( ) {
let me = this ;
if ( in _list ( [ "Delivery Note" , "Sales Invoice" , "Sales Order" ] , this . frm . doc . doctype ) ) {
if ( this . frm . doc . project ) {
frappe . call ( {
method : 'erpnext.projects.doctype.project.project.get_cost_center_name' ,
args : { project : this . frm . doc . project } ,
callback : function ( r , rt ) {
if ( ! r . exc ) {
$ . each ( me . frm . doc [ "items" ] || [ ] , function ( i , row ) {
if ( r . message ) {
frappe . model . set _value ( row . doctype , row . name , "cost_center" , r . message ) ;
frappe . msgprint ( _ _ ( "Cost Center For Item with Item Code {0} has been Changed to {1}" , [ row . item _name , r . message ] ) ) ;
}
} )
}
}
} )
}
}
}
} ;
}
}
erpnext . pre _sales = {
set _as _lost : function ( doctype ) {
frappe . ui . form . on ( doctype , {
set _as _lost _dialog : function ( frm ) {
var dialog = new frappe . ui . Dialog ( {
title : _ _ ( "Set as Lost" ) ,
fields : [
{
"fieldtype" : "Table MultiSelect" ,
"label" : _ _ ( "Lost Reasons" ) ,
"fieldname" : "lost_reason" ,
"options" : frm . doctype === 'Opportunity' ? 'Opportunity Lost Reason Detail' : 'Quotation Lost Reason Detail' ,
"reqd" : 1
} ,
{
"fieldtype" : "Table MultiSelect" ,
"label" : _ _ ( "Competitors" ) ,
"fieldname" : "competitors" ,
"options" : "Competitor Detail"
} ,
{
"fieldtype" : "Small Text" ,
"label" : _ _ ( "Detailed Reason" ) ,
"fieldname" : "detailed_reason"
} ,
] ,
primary _action : function ( ) {
let values = dialog . get _values ( ) ;
frm . call ( {
doc : frm . doc ,
method : 'declare_enquiry_lost' ,
args : {
'lost_reasons_list' : values . lost _reason ,
'competitors' : values . competitors ? values . competitors : [ ] ,
'detailed_reason' : values . detailed _reason
} ,
callback : function ( r ) {
dialog . hide ( ) ;
frm . reload _doc ( ) ;
} ,
} ) ;
} ,
primary _action _label : _ _ ( 'Declare Lost' )
} ) ;
dialog . show ( ) ;
}
} ) ;
}
}