As we all know that all server calls from lightning components are asynchronous in nature. If we call 2 different apex methods from lightning components, then there is no surety that which will return response first. In order to provide sequencing between these 2 calls, we have to write second method call in callback function of first apex call as mentioned below:
Now if you have to perform multiple server side apex calls from lightning components in sequential order, code becomes very difficult to understand and read and difficult to maintain this kind of code.
Javascript Promises
Promise pattern is very common in javascript to handle asynchronous operations. Prior to promises events and callback functions were used but they had limited functionalities and created unmanageable code.
Promise can have 3 states:
Use below syntax to create promise:
Remember:
findDataUsingNormalCall : function(component, event, helper) {
var actionName1= component.get("c.findMyAccounts");
var params1={"numberOfRecords":2};
actionName1.setParams(params1);
actionName1.setCallback(this, function(response) {
var state = response.getState();
if (state === "SUCCESS") {
var apexResponse1=response.getReturnValue();
component.set("v.ltngAccList",apexResponse1);
//Now perform second Call
var actionName2= component.get("c.findMyPendingTasks");
var params2={"numberOfRecords":2};
actionName2.setParams(params2);
actionName2.setCallback(this, function(response) {
state = response.getState();
if (state === "SUCCESS") {
var apexResponse2=response.getReturnValue();
component.set("v.ltngTaskList",apexResponse2);
}else if(state === "ERROR"){
var errors2 = response.getError();
console.error(errors2);
}
});
$A.enqueueAction(actionName2);
}else if(state === "ERROR"){
var errors1 = response.getError();
console.error(errors1);
}
});
$A.enqueueAction(actionName1);
Now if you have to perform multiple server side apex calls from lightning components in sequential order, code becomes very difficult to understand and read and difficult to maintain this kind of code.
Javascript Promises
Promise pattern is very common in javascript to handle asynchronous operations. Prior to promises events and callback functions were used but they had limited functionalities and created unmanageable code.
Promise can have 3 states:
- Pending
- Fulfilled
- Rejected
var promise1 = new Promise($A.getCallback(function(resolve, reject){
//perform logic like server call
if (/* success */) {
resolve("result");
}else {
reject("error");
}
}));
- Constructor takes only one argument as a function.
- Callback function takes two arguments, resolve and reject
- Perform operations inside the callback function and if everything went well then call resolve.
- If desired operations do not go well then call reject.
In order to consume promise, use .then or .catch methods as shown below:
promise1 .
.then($A.getCallback(function(result){
//handle success
}),
$A.getCallback(function(error){
//handle error
})
)
.catch($A.getCallback(function () {
console.log('Some error has occured');
}));
Remember:
- then() method automatically invoked when promise is either resolved (fulfilled) or reject.
- then() method takes 2 functions as parameters:
- If promise is resolved and a result is received, First function is executed.
- If promise is rejected and an error is received, Second function is executed(optional).
- catch() method is invoked when a promise is either rejected or some error has occurred in execution. catch take 1 function as parameter. If you are using the second parameters for then function then use catch for error handling.
Calling multiple asynchronous functions and Chaining them using Promises
You can use below pattern in order to perform different asynchronous operation in synchronous manner.
new Promise($A.getCallback(function(resolve, reject){}))
.then(
// resolve handler
$A.getCallback(function(result) {
//when first call is successfull, then create new
//instance of promise to make another call
return new Promise($A.getCallback(function(resolve,reject){}));
}),
// reject handler
$A.getCallback(function(error) {
console.log("Promise was rejected: ", error);
//you can create another promise for error handling
})
)
.then(
// resolve handler
$A.getCallback(function() {
//perform logic when second call is successfull
})
);
I have created a sample lightning component which explain the functionality of callback function and promise pattern. Below are details about this component functionality:
Below is code snippet:
- Component contains 2 button which invoke 2 different apex methods by using callback pattern and promise pattern.
- When user clicks on "Fetch 2 records using callback pattern" button, system fires 2 server calls to get account and task records in asynchronous manner. So records will get displayed on UI based on response from server. There will be no sequencing of these 2 method invocation.
- When user clicks on "Fetch 3 records using promise pattern", system first fire an asynchronous call to get account records and once account data is recieved, then it will fire another asynchronous call to get task records. This functionality uses promise pattern to fire 2 asynchronous call in synchronous manner.
Below is code snippet:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<aura:component controller="SK_PromisePatternDemoController"> | |
<aura:attribute name="ltngAccList" type="List"/> | |
<aura:attribute name="ltngTaskList" type="List"/> | |
<aura:attribute name="ltngClickedAction" type="String" default=""/> | |
<aura:attribute name="ltngAccInfoString" type="String" default=""/> | |
<aura:attribute name="ltngTaskInfoString" type="String" default=""/> | |
<lightning:button label="Fetch 2 records using callback pattern" onclick="{!c.findDataUsingNormalCall}"/> | |
<lightning:button label="Fetch 3 records using promise pattern" onclick="{!c.findDataUsingPromise}"/> | |
<br/> | |
<aura:if isTrue="{!v.ltngClickedAction=='NormalCall'}"> | |
{!v.ltngAccInfoString} | |
<aura:if isTrue="{!!empty(v.ltngAccList)}"> | |
<div style="background-color:#E6E6FA;border-style: solid;margin:5px;"> | |
<table class="slds-table slds-table_fixed-layout slds-table_bordered slds-table_cell-buffer"> | |
<thead> | |
<tr class="slds-text-title--caps"> | |
<th scope="col">Name</th> | |
<th scope="col">Type</th> | |
<th scope="col">Industry</th> | |
</tr> | |
</thead> | |
<tbody> | |
<aura:iteration items="{!v.ltngAccList}" var="item"> | |
<tr class="slds-hint-parent"> | |
<td scope="row"> {!item.Name}</td> | |
<td > {!item.Type}</td> | |
<td > {!item.Industry}</td> | |
</tr> | |
</aura:iteration> | |
</tbody> | |
</table> | |
</div> | |
</aura:if> | |
<br/> | |
{!v.ltngTaskInfoString} | |
<aura:if isTrue="{!!empty(v.ltngTaskList)}"> | |
<div style="background-color:#E6E6FA;border-style: solid;margin:5px;"> | |
<table class="slds-table slds-table_fixed-layout slds-table_bordered slds-table_cell-buffer"> | |
<thead> | |
<tr class="slds-text-title--caps"> | |
<th scope="col">Subject</th> | |
<th scope="col">Status</th> | |
<th scope="col">Description</th> | |
</tr> | |
</thead> | |
<tbody> | |
<aura:iteration items="{!v.ltngTaskList}" var="item"> | |
<tr class="slds-hint-parent"> | |
<td scope="row"> {!item.Subject}</td> | |
<td > {!item.Status}</td> | |
<td > {!item.Description}</td> | |
</tr> | |
</aura:iteration> | |
</tbody> | |
</table> | |
</div> | |
</aura:if> | |
</aura:if> | |
<br/> | |
<aura:if isTrue="{!v.ltngClickedAction=='PromiseCall'}"> | |
{!v.ltngAccInfoString} | |
<aura:if isTrue="{!!empty(v.ltngAccList)}"> | |
<div style="background-color:#E6E6FA;border-style: solid;margin:5px;"> | |
<table class="slds-table slds-table_fixed-layout slds-table_bordered slds-table_cell-buffer"> | |
<thead> | |
<tr class="slds-text-title--caps"> | |
<th scope="col">Name</th> | |
<th scope="col">Type</th> | |
<th scope="col">Industry</th> | |
</tr> | |
</thead> | |
<tbody> | |
<aura:iteration items="{!v.ltngAccList}" var="item"> | |
<tr class="slds-hint-parent"> | |
<td scope="row"> {!item.Name}</td> | |
<td > {!item.Type}</td> | |
<td > {!item.Industry}</td> | |
</tr> | |
</aura:iteration> | |
</tbody> | |
</table> | |
</div> | |
</aura:if> | |
<br/> | |
{!v.ltngTaskInfoString} | |
<aura:if isTrue="{!!empty(v.ltngTaskList)}"> | |
<div style="background-color:#E6E6FA;border-style: solid;margin:5px;"> | |
<table class="slds-table slds-table_fixed-layout slds-table_bordered slds-table_cell-buffer"> | |
<thead> | |
<tr class="slds-text-title--caps"> | |
<th scope="col">Subject</th> | |
<th scope="col">Status</th> | |
<th scope="col">Description</th> | |
</tr> | |
</thead> | |
<tbody> | |
<aura:iteration items="{!v.ltngTaskList}" var="item"> | |
<tr class="slds-hint-parent"> | |
<td scope="row"> {!item.Subject}</td> | |
<td > {!item.Status}</td> | |
<td > {!item.Description}</td> | |
</tr> | |
</aura:iteration> | |
</tbody> | |
</table> | |
</div> | |
</aura:if> | |
</aura:if> | |
</aura:component> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<aura:application extends="force:slds"> | |
<c:SK_PromisePatternDemo/> | |
</aura:application> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public class SK_PromisePatternDemoController { | |
@AuraEnabled | |
public static List<Account> findMyAccounts(integer numberOfRecords){ | |
return [select id,Name,Type,Industry from Account where createdById=:UserInfo.getUserId() order by LastModifieddate DESC Limit :numberOfRecords]; | |
} | |
@AuraEnabled | |
public static List<Task> findMyPendingTasks(integer numberOfRecords){ | |
return [select id,Description,Status,Subject from Task where OwnerId=:UserInfo.getUserId() order by LastModifieddate DESC Limit :numberOfRecords ]; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
({ | |
findDataUsingNormalCall : function(component, event, helper) { | |
component.set("v.ltngClickedAction","NormalCall"); | |
component.set("v.ltngAccInfoString",""); | |
component.set("v.ltngTaskInfoString",""); | |
var accList=component.set("v.ltngAccList",[]); | |
var accList=component.set("v.ltngTaskList",[]); | |
//find account list | |
component.set("v.ltngAccInfoString","Waiting for apex response from Server to get account records"); | |
var actionName= component.get("c.findMyAccounts"); | |
var params={"numberOfRecords":2}; | |
actionName.setParams(params); | |
actionName.setCallback(this, function(response) { | |
var state = response.getState(); | |
if (state === "SUCCESS") { | |
var apexResponse=response.getReturnValue(); | |
//set response to attribute value | |
console.log('***'+JSON.stringify(apexResponse)); | |
component.set("v.ltngAccInfoString","Response recieved from Server for Account records."); | |
component.set("v.ltngAccList",apexResponse); | |
}else if(state === "ERROR"){ | |
var errors = response.getError(); | |
console.error(errors); | |
alert('Problem with connection. Contact your system administrator.'); | |
} | |
}); | |
$A.enqueueAction(actionName); | |
//find task list | |
component.set("v.ltngTaskInfoString","Waiting for apex response from Server to get task records"); | |
var actionName= component.get("c.findMyPendingTasks"); | |
var params={"numberOfRecords":2}; | |
actionName.setParams(params); | |
actionName.setCallback(this, function(response) { | |
var state = response.getState(); | |
if (state === "SUCCESS") { | |
var apexResponse=response.getReturnValue(); | |
//set response to attribute value | |
console.log('***'+JSON.stringify(apexResponse)); | |
component.set("v.ltngTaskInfoString","Response recieved from Server for Task records."); | |
component.set("v.ltngTaskList",apexResponse); | |
}else if(state === "ERROR"){ | |
var errors = response.getError(); | |
console.error(errors); | |
alert('Problem with connection. Contact your system administrator.'); | |
} | |
}); | |
$A.enqueueAction(actionName); | |
}, | |
findDataUsingPromise : function (component,event,helper){ | |
console.log('****inside findDataUsingPromise'); | |
component.set("v.ltngAccInfoString",""); | |
component.set("v.ltngTaskInfoString",""); | |
component.set("v.ltngClickedAction","PromiseCall"); | |
var accList=component.set("v.ltngAccList",[]); | |
var accList=component.set("v.ltngTaskList",[]); | |
component.set("v.ltngAccInfoString","Waiting for response from Server to get Account Records."); | |
var params={"numberOfRecords":3}; | |
var accountDataPromise = helper.ServerCallUsingPromise(component,'c.findMyAccounts',params); | |
accountDataPromise | |
.then( | |
$A.getCallback(function(result){ | |
// Set Account Attribute | |
console.log('***inside success for first promise call'); | |
console.log('***'+JSON.stringify(result)); | |
component.set("v.ltngAccInfoString","Response recieved from Server for Account Records."); | |
component.set("v.ltngAccList", result); | |
console.log('***calling pending task using second promise'); | |
var params={"numberOfRecords":3}; | |
component.set("v.ltngTaskInfoString","Waiting for response from Server to get Task Records."); | |
return helper.ServerCallUsingPromise(component,'c.findMyPendingTasks',params); | |
}), | |
$A.getCallback(function(error){ | |
alert('An error occurred getting the account : ' + error.message); | |
}) | |
) | |
.then( | |
$A.getCallback(function(result){ | |
console.log('***inside success for second promise call'); | |
console.log('***'+JSON.stringify(result)); | |
component.set("v.ltngTaskInfoString","Response recieved from Server for task records."); | |
component.set("v.ltngTaskList", result); // Set task Attribute | |
}), | |
$A.getCallback(function(error){ | |
// Something went wrong | |
alert('An error occurred while getting the task : ' + error.message); | |
}) | |
); | |
} | |
}) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
({ | |
callToServer : function(component, method, callback, params) { | |
var action = component.get(method); | |
if(params){ | |
action.setParams(params); | |
} | |
console.log('****param to controller:'+JSON.stringify(params)); | |
action.setCallback(this, function(response) { | |
var state = response.getState(); | |
if (state === "SUCCESS") { | |
callback.call(this,response.getReturnValue()); | |
}else if(state === "ERROR"){ | |
var errors = response.getError(); | |
console.error(errors); | |
alert('Problem with connection.'+errors); | |
} | |
}); | |
$A.enqueueAction(action); | |
}, | |
ServerCallUsingPromise : function( component,method, params ) { | |
var promiseInstance = new Promise( $A.getCallback( function( resolve , reject ) { | |
var action = component.get(method); | |
if(params){ | |
action.setParams(params); | |
} | |
console.log('****param to controller:'+JSON.stringify(params)); | |
action.setCallback(this, function(response) { | |
var state = response.getState(); | |
if (state === "SUCCESS") { | |
resolve(response.getReturnValue()); | |
}else if(state === "ERROR"){ | |
var errors = response.getError(); | |
console.error(errors); | |
reject(response.getError() ); | |
} | |
}); | |
$A.enqueueAction(action); | |
})); | |
return promiseInstance; | |
} | |
}) |
Best Practices:
- Always use catch or reject handler.
- Always use $A.getCallback() when using promise pattern in lightning components. Even if you do not use this, then sometimes it will work but it will be very difficult to debug if something went wrong. Sometimes if you don't use it, then results may get delayed on UI and can cause performance issues.
Hope this will help!!
Do you need Finance? Are you looking for Finance? Are you looking for finance to enlarge your business? We help individuals and companies to obtain finance for business expanding and to setup a new business ranging any amount. Get finance at affordable interest rate of 3%, Do you need this finance for business and to clear your bills? Then send us an email now for more information contact us now via (financialserviceoffer876@gmail.com) whats-App +918929509036 Dr James Eric Finance Pvt Ltd Thanks
ReplyDeleteBuy Tramadol 100mg TOP-DOL 180-Pills
ReplyDeleteBuy Ativan 2mg Online Cheap Price
ReplyDeleteBuy clonazepam 1mg onilne
ReplyDeleteBuy clonazepam 2mg online
ReplyDeleteBuy Ambien Online
ReplyDeleteBuy Tramadol 100 mg Online
ReplyDeleteBuy Ambien Online Overnight
ReplyDeleteOrder Tapentadol 100 Mg Online
ReplyDeleteGeneric cialis 10 mg online
ReplyDeleteAre you interested in any kinds of hacking services?
ReplyDeleteFeel free to contact TECHNECHHACKS.
For years now we’ve helped so many organizations and companies in hacking services.
TECHNECHHACKS is a team of certified hackers that has their own specialties and they are five star rated hackers.
We give out jobs to hackers (gurus only) to those willing to work, with or without a degree, to speed up the availability of time given to jobs!!
Thus an online binary decoding exam will be set for those who needs employment under the teams establishment.
we deal with the total functioning of sites like,
• SOCIAL MEDIA (Facebook, Twitter, Instagram, Snapchat, google hangout etc.)
• SCHOOL GRADES
• IOS/OS
• CREDIT SCORES
• BANK ACCOUNTS
• SPOUSES PHONE
Our special agents are five star rated agents that specializes in the following, and will specially be assigned to you for a special job well DONE.
• WESTERN UNION TRANSFER
• CREDIT CARDS INSTALLATION
• MONEY FLIPPING
• CRIMINAL RECORDS
• BTC RECOVERY
• BTC MINING
• BTC INVESTMENT
Thus bewere of scammers because most persons are been scammed and they ended up getting all solutions to their cyber bullies and attacks by US.
I am Jason williams one of the leading hack agent.
PURPOSE IS TO GET YOUR JOBS DONE AT EXACTLY NEEDED TIME REQUESTED!!!
And our WORK SUCCESS IS 100%!!!
We’re always available for you when you need help.
Contact or write us on:
Technechhacks@gmail.com
SIGNED....!
Jason. W
TECHNECHHACKS
2021©️All Right Reserved
ReplyDeleteBest Pain Reliever Buy Tapentadol Aspadol 100 mg Online
Tapentadol Aspadol 100mg is used for the treatment of moderate to severe pain for both acute (following injury, surgery, etc.) and chronic musculoskeletal pain.
Are you in need of Loan? Here all problem regarding Loans is solve between a short period of time what are you waiting for apply now and solve your problem or start a business with funds Contact us now. many more 2% interest rate.(Whats App) number +919394133968 patialalegitimate515@gmail.com
ReplyDeleteMr Sorina
You decide to Buy Tapentadol 100mg online for benefits.
ReplyDeleteTapentadol 100mg tablets are used to treat moderate to severe acute pain (pain that begins suddenly, has a specific cause,and is expected to go away when the cause of the pain is healed).
Infycle Technology, the finest Software Training Institutes in Chennai Offering technical courses like Big Data, Cyber Security, Artificial Intelligence, Oracle, Java, Hadoop, Selenium, Android, and iOS Development, Data Science etc. It is the Best place to take Big Data training in Chennai for freshers, students, and tech professionals. For more enquiry and the free demo classes, Contact 7504633633.
ReplyDeleteInsomnia is a sleep condition in which you consider it impossible to fall asleep. .buy Ambien Sanval 10mg online It may be short-term or long-term (acute) or chronic. Sleeplessness is chronic if it lasts three nights or longer per week, at least. Decide to for benefits
ReplyDeleteAre you suffering from unbearable muscle and bone pain? Has your doctor or pharmacist prescribed you SOMA for relief from the discomfort and pain associated with the painful, Buy soma 350 mg online acute musculoskeletal condition
ReplyDeleteThis post is so interactive and informative.keep updating more information...
ReplyDeleteAWS Classes In Mumbai
AWS Certification In Ahmedabad
AWS Certification In Kochi
AWS Course In Trivandrum
AWS Course In Kolkata
ReplyDeleteIf that is what one’s life is,
Tramadol 50 mg order online is the answer to your prayers. As effective as it is in relieving pain, it does come with some side effects, so stay alert to symptoms of dizziness, drowsiness, blurred vision, sleep problems, agitation, anxiety, constipation, diarrhea, hallucination, nausea, tremor, vomiting, and diaphoresis.
Buy Ksalol 1mg Online is used to treat anxiety and panic disorders. It works by enhancing the effects of a certain natural chemical in the body. For more details please visit on website...
ReplyDeleteWhatsapp Number: +1 701450155
ReplyDeleteI am very happy to read this article. Thanks for giving us Amazing info. Fantastic post.
Thanks For Sharing such an informative article, Im taking your feed also, Thanks.yodot-rar-repair-crack/
I was looking for this information from enough time and now I reached your website it’s really good content.
ReplyDeleteThanks for writing such a nice content for us.
2020/03/28/windows-7-serial-key
I am very happy to read this article. Thanks for giving us Amazing info. Fantastic post.
ReplyDeleteThanks For Sharing such an informative article, Im taking your feed also, Thanks.
snaptube-crack/
I am very happy to read this article. Thanks for giving us Amazing info. Fantastic post.
ReplyDeleteThanks For Sharing such an informative article, Im taking your feed also, Thanks.yodot key
This comment has been removed by the author.
ReplyDeleteGreat this site and its information is very well. Thanks for sharing information.
ReplyDeleteigoal88 แนะนำเพื่อน
Bitcoin and other cryptocurrency scams are becoming rampant as the day goes by. One should be careful when dealing with people online especially when it has to do with your funds. Getting back your stolen bitcoins is very easy but there are also alots of fraudulent recovery firms out there so you need to be careful not to end up being defrauded again. Jamesmckaywizard is the most trusted and reliable recovery firm online. They are the best recovery firm out there now. They are very fast and efficient in the recovery of your funds. To contact them ON EMAIL:jamesmckaywizard@gmail.com or What's App:+44 7826 613094, You can simply send an email and you will be helped to recover 100% of you lost funds- I was able to recover all of my stolen bitcoins. They provide excellent service
ReplyDeletevery informative blog. Full Stack Classes In Pune
ReplyDeleteNice blog.
ReplyDeleteplease check Java course in Pune
Hiii...Thank you so much for sharing Great information....Nice post....Keep moving on....
ReplyDelete