
Hi there, At the time of this blog post, There is no standard Lightning component available for Email composer, So I created a custom component for sending Email and mimic of the standard email composer and also for sending Listview selected records. I assumed sending email for person accounts (person accounts by default not available in salesforce you have to rise a case to enable them) of Opportunities. Please choose different object for sending(from contact List view) since if you don't have person accounts enabled you will get compile errors. You also need to create a rich text field in any object since we don't have standard richtext tags in lighting at the time of posting this blog post. We need rich text field to mimic the email composer screen and also to render the text review of template selected. I have used a List view button of type visualforce in opportunities List view and in visualforce page i have placed Lightning Component. The list view selected records will be sent to Lightning Component. Get the snippets below to image. Visualforce Page: 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 Show hidden characters <apex:page standardController="Opportunity" action="{!doSomething}" recordSetVar="opportunities" tabStyle="Opportunity" extensions="ListEmaillOppController"> <apex:includeLightning /> <apex:slds /> <style> .close-btn{text-align: right;color: white;font-size: 20px;padding: 0.5%;cursor: pointer;} </style> <script src="https://ajax.googleapis.com/ajax/libs/jquery/3.2.1/jquery.min.js"></script> <script> var myUserContext = "{!$User.UITheme}"; var accIds = "{!accIdStr}"; var oppIds = "{!oppIdStr}"; var visualForceFunction = function(){ if (myUserContext == 'Theme4t' || myUserContext == 'Theme4d') { // The Visualforce page is in S1 or Lightning Experience sforce.one.navigateToURL("/006/o"); } else if (myUserContext == 'Theme3') { // The Visualforce page is running in Classic window.top.location.href = '{!$Label.OrgUrl}'+'006/o'; } else { console.log("Unsupported theme"); } }; $Lightning.use("c:SendListEmailOnOpp", function() { console.log('lead Id', '{!$CurrentPage.parameters.Id}'); $Lightning.createComponent("c:ListEmailOpp",{"accIds" : accIds, "oppIds" : oppIds},"lightning",function(cmp) { $('.close-btn').click(function(){ visualForceFunction(); }); $A.eventService.addHandler({ event: 'force:navigateToURL', handler: function(event) { visualForceFunction(); } }); }); }); </script> <input type="hidden" id="leadHiddenId" value="{!$CurrentPage.parameters.Id}"/> <div class="demo-only" style="height: 640px;"> <section role="dialog" tabindex="-1" aria-labelledby="modal-heading-01" aria-modal="true" aria-describedby="modal-content-id-1" class="slds-modal slds-fade-in-open"> <div class="slds-modal__container"> <span class="close-btn">X</span> <div class="slds-modal__content slds-p-around_medium" id="lightning"> </div> </div> </section> <div class="slds-backdrop slds-backdrop_open"></div> </div> </apex:page> view raw ListEmailPage.xml hosted with ❤ by GitHub Controller Extension: 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 Show hidden characters /* Class name : ListEmaillOppController, Created By : Anil Kumar Description : Enhancements : */ public class ListEmaillOppController { private ApexPages.StandardSetController standardSetController; public string accIdStr{get;set;} public string oppIdStr{get;set;} public ListEmaillOppController(ApexPages.StandardSetController standardSetController) { this.standardSetController = standardSetController; accIdStr=''; } public PageReference doSomething() { // Apex code for handling records from a List View goes here List<Opportunity> selectedListViewRecords = (List<Opportunity>) standardSetController.getSelected(); Boolean hasMore = standardSetController.getHasNext(); if(selectedListViewRecords.size()>0){ list<string> accIds = new list<string>(); list<string> oppIds = new list<string>(); for(Opportunity oppRec : [select id,accountId from Opportunity WHERE Id IN: selectedListViewRecords]){ accIds.add(oppRec.accountId); oppIds.add(oppRec.Id); } accIdStr = accIds.size()>0 ? String.join(accIds, ',') : ''; oppIdStr= oppIds.size()>0 ? String.join(oppIds, ',') : ''; } system.debug('---selectedListViewRecords ---'+selectedListViewRecords.size() ); return null; } } view raw ListEmaillOppController.java hosted with ❤ by GitHub EmailComp: 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 Show hidden characters <aura:component controller="ListEmailOppCntrlr" implements="force:hasRecordId,flexipage:availableForAllPageTypes,force:lightningQuickActionWithoutHeader" access="global"> <aura:handler name="init" value="{!this}" action="{!c.loadComponent}"/> <aura:attribute name="errorMsg" type="String" description=""/> <aura:attribute name="shoMsg" type="boolean" default="false" description=""/> <aura:attribute name="shwSucesMsg" type="boolean" default="false" description=""/> <aura:attribute name="showLoader" type="boolean" default="false" description=""/> <aura:attribute name="Opportunity" type="Opportunity" default="{ 'sobjectType': 'Opportunity', 'SampleRichText__c':''}"/> <aura:attribute name="disableTemplate" type="boolean" default="false" description=""/> <aura:attribute name="selTempl" type="String" description=""/> <aura:attribute name="templates" type="EmailTemplate[]" default="[]"/> <aura:attribute name="addnlEmails" type="String" default="" description=""/> <aura:attribute name="subjTxt" type="String" default="" description=""/> <aura:attribute name="msgReview" type="String" default="" description=""/> <aura:attribute name="accIds" type="String" access="GLOBAL" default="" description=""/> <aura:attribute name="oppIds" type="String" access="GLOBAL" default="" description=""/> <aura:attribute name="accRecords" type="sObject[]" default="[]" /> <aura:attribute name="templDetail" type="EmailTemplate" default="{}" /> <div class="slds-page-header" role="banner"> <div class="slds-grid"> <div class="slds-col slds-has-flexi-truncate"> <!-- SEARCH AREA --> <p class="slds-text-title--caps slds-line-height--reset"> <span title="lead Standard Icon"> </span> Send Email</p> <!-- / SEARCH AREA --> </div> </div> </div> <br/><br/> <!-- Connection 1 Section --> <div class="slds-grid slds-grid_vertical"> <div class="slds-box slds-box--small slds-theme--shade slds-text-align--left">Email Message</div> <br/> <div class="slds-col"> <div class="slds-form-element slds-size--1-of-1"> <!-- <label class="slds-form-element__label" for="input-02">Recipients<span color="red"><b>*</b></span></label> --> <!-- <input type="Text" autocomplete="off" style="border-left:4px solid red;" value="" class="slds-input" /> --> <c:reUsableMultiSelectLookup objectAPIName="account" IconName="standard:account" lstSelectedRecords="{!v.accRecords}" label="Recipients" aura:id="lookupRes" /> </div> </div> <br/><br/> <div class="slds-col"> <lightning:select disabled="{!v.disableTemplate}" onchange="{!c.loadTemplate}" name="SelectDivision" label="Select a Template:" aura:id="templateId" value="{!v.selTempl}"> <option text="None" value=""/> <aura:iteration items="{!v.templates}" var="item"> <option text="{!item.Name}" value="{!item.Id}"/> </aura:iteration> </lightning:select> </div> <br/><br/> <div class="slds-col"> <div class="slds-form-element slds-size--1-of-1"> <label class="slds-form-element__label" for="input-02">Subject:</label> <input id="subjMatter" type="Text" autocomplete="off" value="{!v.subjTxt}" class="slds-input" /> </div> </div><br/><br/> <div class="slds-col" aura:id="emailBodyDiv"> <div class="slds-form-element slds-size--1-of-1"> <label class="slds-form-element__label" for="input-02">Content :</label> <span></span> <force:inputField class="richTxt" value="{!v.Opportunity.SampleRichText__c}"/> </div> </div><br/><br/> <aura:if isTrue="{!!empty(v.selTempl)}" > <lightning:textarea name="myTextArea" value="{!v.templDetail.Body}" label="Content:" maxlength="700" class="txtAreaCls" disabled="true" /> <br/><br/> </aura:if> </div> <br/> <!-- Error message area --> <br/><br/> <aura:if isTrue="{!v.shoMsg}" > <ui:message aura:id="errPanel" title="Error" severity="error" closable="false"> {!v.errorMsg} </ui:message> <br/><br/> </aura:if> <aura:if isTrue="{!v.shwSucesMsg}" > <ui:message aura:id="errPanel" title="Success!" severity="confirm" closable="false"> Email has been sent! </ui:message> <br/><br/> </aura:if> <!-- Buttons and Loader section --> <div class="slds-align_absolute-center " style="padding:2%;"> <lightning:button variant="brand" label="Send Email" onclick="{!c.sendEmailAction}" /> <input type="button" value="Cancel" class="slds-button slds-button--neutral" onclick="{!c.closeDialog}"/> <aura:if isTrue="{!v.showLoader}"> <div class="demo-only" style="height: 6rem;"> <div role="status" class="slds-spinner_brand slds-spinner slds-spinner_small"> <span class="slds-assistive-text">Loading</span> <div class="slds-spinner__dot-a"></div> <div class="slds-spinner__dot-b"></div> </div> </div> </aura:if> </div> </aura:component> view raw ListEmailComp.xml hosted with ❤ by GitHub Controller.js: 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 Show hidden characters ({ loadComponent : function(component, event, helper) { helper.getSelctedAccountsOfOpportunity(component, event); helper.getEmailTempaltes(component, event); console.log(component.get("v.accIds")); }, sendEmailAction : function(component, event, helper) { helper.sendEmails(component, event); }, loadTemplate : function(component, event, helper) { helper.getTemplate(component, event); }, closeDialog : function(component, event, helper) { helper.cancelAction(component, event); } }) view raw ListEmailCompController.js hosted with ❤ by GitHub Helper.js: 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 Show hidden characters ({ getEmailTempaltes : function(component, event) { var action = component.get("c.getTemplates"); //action.setParams({"divisionId":selectedDivision}); action.setCallback(this,function(response){ var loadResponse = response.getReturnValue(); console.log('templates..!',loadResponse); if(!$A.util.isEmpty(loadResponse)){ component.set('v.templates',loadResponse); } }); $A.enqueueAction(action); }, getSelctedAccountsOfOpportunity : function(component, event) { var accIdsofOpp = component.get("v.accIds"); if(!$A.util.isEmpty(accIdsofOpp)){ var action = component.get("c.getAccountsOfOpportunity"); action.setParams({"accIds":accIdsofOpp}); action.setCallback(this,function(response){ var responseVal = response.getReturnValue(); console.log('responseVal..** ',responseVal); if(!$A.util.isEmpty(responseVal)){ component.set("v.accRecords",responseVal); } }); $A.enqueueAction(action); } }, sendEmails : function(component, event) { component.set("v.shoMsg", false); component.set("v.showLoader", true); var selRec = component.find("lookupRes").get("v.lstSelectedRecords"); var templateId = component.get("v.selTempl"); var oppRecIds = component.get("v.oppIds"); console.log('oppRecIds ', oppRecIds); console.log('sel records ', selRec); console.log('sel template ', templateId); //var addnlEmails = document.getElementById("addnlEmail").value; var subjMatter = document.getElementById("subjMatter").value; console.log('subjMatter ',subjMatter); var emailBody = !$A.util.isEmpty(component.get("v.Opportunity").SampleRichText__c) ? component.get("v.Opportunity").SampleRichText__c : ''; console.log('emailBody ',emailBody); if(!$A.util.isEmpty(selRec) && (!$A.util.isEmpty(emailBody) || !$A.util.isEmpty(templateId)) ){ var accIds = []; for (var i = 0; i<selRec.length; i++) { accIds.push(selRec[i].PersonContactId); } console.log('---accIds--- ', accIds); var accIdStr = !$A.util.isEmpty(accIds) ? accIds.join() : ''; console.log('---accIdStr--- ', accIdStr); if(!$A.util.isEmpty(accIdStr) || !$A.util.isEmpty(addnlEmails)){ console.log('---accIdStr--- ', accIdStr); console.log('--Opportunity--', component.get("v.Opportunity")); var action = component.get("c.sendAnEmailMsg"); action.setParams({"templateId":templateId, "accIds":!$A.util.isEmpty(accIdStr) ? accIdStr : '', "opty":component.get("v.Opportunity"), "subj" : !$A.util.isEmpty(subjMatter) ? subjMatter : '', "addnlEmails" : '', "oppIds" : oppRecIds }); action.setCallback(this,function(response){ var emailMsgResp = response.getReturnValue(); console.log('--emailMsgResp--', emailMsgResp); //isSuccess errMsg component.set("v.showLoader", false); if(emailMsgResp.isSuccess){ component.set("v.shwSucesMsg", true); this.cancelAction(component, event); } else { component.set("v.shoMsg", true); component.set("v.errorMsg", emailMsgResp.errMsg); } }); $A.enqueueAction(action); } } else { component.set("v.showLoader", false); component.set("v.shoMsg", true); component.set("v.errorMsg", "Please provide Recipient, Template or Email Body"); } }, getTemplate : function(component, event) { var templId = component.get("v.selTempl"); component.set("v.showLoader", true); if(!$A.util.isEmpty(templId)){ var action = component.get("c.getTemplateDetails"); action.setParams({"templteId":templId}); action.setCallback(this,function(response){ component.set("v.showLoader", false); var responseVal = response.getReturnValue(); console.log('responseVal..@getTemplate ',responseVal); if(!$A.util.isEmpty(responseVal)){ component.set("v.templDetail",responseVal); component.set("v.subjTxt",responseVal.Subject); if(!$A.util.hasClass(component.find("emailBodyDiv"), "slds-hide")){ $A.util.addClass(component.find("emailBodyDiv"), 'slds-hide'); } } }); $A.enqueueAction(action); } else { component.set("v.showLoader", false); if($A.util.hasClass(component.find("emailBodyDiv"), "slds-hide")){ $A.util.removeClass(component.find("emailBodyDiv"), 'slds-hide'); } } }, cancelAction: function(component, event){ var urlEvent = $A.get("e.force:navigateToURL"); urlEvent.setParams({ "url": '/006/o' }); urlEvent.fire() } }) view raw ListEmailCompHelper.js hosted with ❤ by GitHub EmailComp Apex Class: 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 Show hidden characters public class ListEmailOppCntrlr { @AuraEnabled public static list<EmailTemplate> getTemplates(){ //RelatedEntityType='Account' AND list<EmailTemplate> emailTemp = new list<EmailTemplate>(); emailTemp = [SELECT Id,Name,Subject,TemplateType FROM EmailTemplate WHERE TemplateType IN ('custom','text')]; return emailTemp; } @AuraEnabled public static list<Account> getAccountsOfOpportunity(string accIds){ list<string> accIdsList = string.isNotBlank(accIds) ? accIds.split(',') : new list<string>(); list<Account> accRecords = accIdsList.size()>0 ? [SELECT Id,Name,PersonContactId FROM Account WHERE ID IN:accIdsList] : new list<Account>(); return accRecords; } @AuraEnabled public static EmailTemplate getTemplateDetails(string templteId){ EmailTemplate emailTemp = new EmailTemplate(); list<EmailTemplate> emailTempLst = new list<EmailTemplate>(); emailTempLst = [SELECT Id,Name,Subject,TemplateType,body FROM EmailTemplate WHERE ID=: templteId]; emailTemp = emailTempLst.size()>0 ? emailTempLst[0] : emailTemp; return emailTemp; } @AuraEnabled public static sucesMsg sendAnEmailMsg(string templateId, string accIds, opportunity opty, string subj, string addnlEmails, string oppIds){ try { List<Messaging.SingleEmailMessage> mails = new List<Messaging.SingleEmailMessage>(); Map<Id,Account> accMap = new map<Id,Account>(); list<Opportunity> oppList = new list<Opportunity>(); list<string> accIdList = string.isNotBlank(accIds) ? accIds.split(',') : new list<string>(); list<string> oppIdList = string.isNotBlank(oppIds) ? oppIds.split(',') : new list<string>(); list<string> addnlEmailList = string.isNotBlank(addnlEmails) ? addnlEmails.split(',') : new list<string>(); system.debug('---accIdList-- '+accIdList); system.debug('---templateId--- '+templateId); system.debug('---oppIdList--- '+oppIdList); if(accIdList.size()>0 && oppIdList.size()>0) { accMap = new Map<Id,Account>([select id,name,personEmail,PersonContactId from Account WHERE PersonContactId IN: accIdList]); oppList = [select id,AccountId from Opportunity WHERE ID IN: oppIdList]; for(opportunity oppRec : oppList) { if(oppRec.accountId!=null && accMap.containsKey(oppRec.AccountId)) { Account persnAcc = accMap.get(oppRec.AccountId); if (persnAcc.personEmail != null) { Messaging.SingleEmailMessage mail = new Messaging.SingleEmailMessage(); if(string.isNotBlank(templateId)){ mail.setTemplateId(ID.valueOf(templateId));//00XW0000000MjvI 00XW0000000MjuyMAC mail.setTargetObjectId(persnAcc.PersonContactId); mail.saveAsActivity=true; mail.setWhatId(oppRec.id); //006W000000DYyeL } else if(string.isNotBlank(opty.SampleRichText__c)){ mail.setSubject(string.isNotBlank(subj) ? subj : ''); List<String> sendTo = new List<String>(); sendTo.add(persnAcc.personEmail); mail.setToAddresses(sendTo); mail.setHtmlBody(opty.SampleRichText__c); mail.saveAsActivity=true; mail.setWhatId(oppRec.id); } // Step 5. Add your email to the master list mails.add(mail); } } } } if(mails.size()>0){ try { Messaging.sendEmail(mails); return new sucesMsg(true, 'Sucess!'); } catch(Exception e) { system.debug('*************--Exception @SendEmail method ---*********'); System.debug('****--Exception type caught: ' + e.getTypeName()); System.debug('****--Message: ' + e.getMessage()); System.debug('****--Cause: ' + e.getCause()); System.debug('****--Line number: ' + e.getLineNumber()); System.debug('****--Stack trace: ' + e.getStackTraceString()); return new sucesMsg(false, e.getMessage()+' - '+e.getCause()+' - '+e.getStackTraceString()+' - '+e.getLineNumber()); } } return new sucesMsg(true, 'Sucess!'); } catch(Exception e) { system.debug('*************--Exception @sendAnEmailMsg method ---*********'); System.debug('****--Exception type caught: ' + e.getTypeName()); System.debug('****--Message: ' + e.getMessage()); System.debug('****--Cause: ' + e.getCause()); System.debug('****--Line number: ' + e.getLineNumber()); System.debug('****--Stack trace: ' + e.getStackTraceString()); return new sucesMsg(false, e.getMessage()+' - '+e.getCause()+' - '+e.getStackTraceString()+' - '+e.getLineNumber()); } } public class sucesMsg { @AuraEnabled public boolean isSuccess; @AuraEnabled public string errMsg; public sucesMsg(boolean isSuccess, string errMsg){ this.isSuccess = isSuccess; this.errMsg = errMsg; } } } view raw ListEmailOppCntrlr.java hosted with ❤ by GitHub I did not mentioned the snippet for the component "c:reUsableMultiSelectLookup" here. Because it is a custom look up component that can be available from many places like salesforce forums and blogs. All you need to do is to tweak the init section of your look up component by passing the selected items to display them automatically. ...